File: //home/arjun/projects/buyercall/node_modules/@ckeditor/ckeditor5-widget/src/widgetresize/resizer.js
/**
* @license Copyright (c) 2003-2022, CKSource Holding sp. z o.o. All rights reserved.
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
*/
/**
* @module widget/widgetresize/resizer
*/
import Template from '@ckeditor/ckeditor5-ui/src/template';
import Rect from '@ckeditor/ckeditor5-utils/src/dom/rect';
import compareArrays from '@ckeditor/ckeditor5-utils/src/comparearrays';
import ObservableMixin from '@ckeditor/ckeditor5-utils/src/observablemixin';
import mix from '@ckeditor/ckeditor5-utils/src/mix';
import ResizeState from './resizerstate';
import SizeView from './sizeview';
/**
* Represents a resizer for a single resizable object.
*
* @mixes module:utils/observablemixin~ObservableMixin
*/
export default class Resizer {
/**
* @param {module:widget/widgetresize~ResizerOptions} options Resizer options.
*/
constructor( options ) {
/**
* Stores the state of the resizable host geometry, such as the original width, the currently proposed height, etc.
*
* Note that a new state is created for each resize transaction.
*
* @readonly
* @member {module:widget/widgetresize/resizerstate~ResizerState} #state
*/
/**
* A view displaying the proposed new element size during the resizing.
*
* @protected
* @readonly
* @member {module:widget/widgetresize/sizeview~SizeView} #_sizeView
*/
/**
* Options passed to the {@link #constructor}.
*
* @private
* @type {module:widget/widgetresize~ResizerOptions}
*/
this._options = options;
/**
* A wrapper that is controlled by the resizer. This is usually a widget element.
*
* @private
* @type {module:engine/view/element~Element|null}
*/
this._viewResizerWrapper = null;
/**
* The width of the resized {@link module:widget/widgetresize~ResizerOptions#viewElement viewElement} before the resizing started.
*
* @private
* @member {Number|String|undefined} #_initialViewWidth
*/
/**
* @observable
*/
this.set( 'isEnabled', true );
this.decorate( 'begin' );
this.decorate( 'cancel' );
this.decorate( 'commit' );
this.decorate( 'updateSize' );
this.on( 'commit', event => {
// State might not be initialized yet. In this case, prevent further handling and make sure that the resizer is
// cleaned up (#5195).
if ( !this.state.proposedWidth && !this.state.proposedWidthPercents ) {
this._cleanup();
event.stop();
}
}, { priority: 'high' } );
this.on( 'change:isEnabled', () => {
// We should redraw the resize handles when the plugin is enabled again.
// Otherwise they won't show up.
if ( this.isEnabled ) {
this.redraw();
}
} );
}
/**
* Attaches the resizer to the DOM.
*/
attach() {
const that = this;
const widgetElement = this._options.viewElement;
const editingView = this._options.editor.editing.view;
editingView.change( writer => {
const viewResizerWrapper = writer.createUIElement( 'div', {
class: 'ck ck-reset_all ck-widget__resizer'
}, function( domDocument ) {
const domElement = this.toDomElement( domDocument );
that._appendHandles( domElement );
that._appendSizeUI( domElement );
that.on( 'change:isEnabled', ( evt, propName, newValue ) => {
domElement.style.display = newValue ? '' : 'none';
} );
domElement.style.display = that.isEnabled ? '' : 'none';
return domElement;
} );
// Append the resizer wrapper to the widget's wrapper.
writer.insert( writer.createPositionAt( widgetElement, 'end' ), viewResizerWrapper );
writer.addClass( 'ck-widget_with-resizer', widgetElement );
this._viewResizerWrapper = viewResizerWrapper;
} );
}
/**
* Starts the resizing process.
*
* Creates a new {@link #state} for the current process.
*
* @fires begin
* @param {HTMLElement} domResizeHandle Clicked handle.
*/
begin( domResizeHandle ) {
this.state = new ResizeState( this._options );
this._sizeView._bindToState( this._options, this.state );
this._initialViewWidth = this._options.viewElement.getStyle( 'width' );
this.state.begin( domResizeHandle, this._getHandleHost(), this._getResizeHost() );
}
/**
* Updates the proposed size based on `domEventData`.
*
* @fires updateSize
* @param {Event} domEventData
*/
updateSize( domEventData ) {
const newSize = this._proposeNewSize( domEventData );
const editingView = this._options.editor.editing.view;
editingView.change( writer => {
const unit = this._options.unit || '%';
const newWidth = ( unit === '%' ? newSize.widthPercents : newSize.width ) + unit;
writer.setStyle( 'width', newWidth, this._options.viewElement );
} );
// Get an actual image width, and:
// * reflect this size to the resize wrapper
// * apply this **real** size to the state
const domHandleHost = this._getHandleHost();
const domHandleHostRect = new Rect( domHandleHost );
newSize.handleHostWidth = Math.round( domHandleHostRect.width );
newSize.handleHostHeight = Math.round( domHandleHostRect.height );
// Handle max-width limitation.
const domResizeHostRect = new Rect( domHandleHost );
newSize.width = Math.round( domResizeHostRect.width );
newSize.height = Math.round( domResizeHostRect.height );
this.redraw( domHandleHostRect );
this.state.update( newSize );
}
/**
* Applies the geometry proposed with the resizer.
*
* @fires commit
*/
commit() {
const unit = this._options.unit || '%';
const newValue = ( unit === '%' ? this.state.proposedWidthPercents : this.state.proposedWidth ) + unit;
// Both cleanup and onCommit callback are very likely to make view changes. Ensure that it is made in a single step.
this._options.editor.editing.view.change( () => {
this._cleanup();
this._options.onCommit( newValue );
} );
}
/**
* Cancels and rejects the proposed resize dimensions, hiding the UI.
*
* @fires cancel
*/
cancel() {
this._cleanup();
}
/**
* Destroys the resizer.
*/
destroy() {
this.cancel();
}
/**
* Redraws the resizer.
*
* @param {module:utils/dom/rect~Rect} [handleHostRect] Handle host rectangle might be given to improve performance.
*/
redraw( handleHostRect ) {
const domWrapper = this._domResizerWrapper;
// Refresh only if resizer exists in the DOM.
if ( !existsInDom( domWrapper ) ) {
return;
}
const widgetWrapper = domWrapper.parentElement;
const handleHost = this._getHandleHost();
const resizerWrapper = this._viewResizerWrapper;
const currentDimensions = [
resizerWrapper.getStyle( 'width' ),
resizerWrapper.getStyle( 'height' ),
resizerWrapper.getStyle( 'left' ),
resizerWrapper.getStyle( 'top' )
];
let newDimensions;
if ( widgetWrapper.isSameNode( handleHost ) ) {
const clientRect = handleHostRect || new Rect( handleHost );
newDimensions = [
clientRect.width + 'px',
clientRect.height + 'px',
undefined,
undefined
];
}
// In case a resizing host is not a widget wrapper, we need to compensate
// for any additional offsets the resize host might have. E.g. wrapper padding
// or simply another editable. By doing that the border and resizers are shown
// only around the resize host.
else {
newDimensions = [
handleHost.offsetWidth + 'px',
handleHost.offsetHeight + 'px',
handleHost.offsetLeft + 'px',
handleHost.offsetTop + 'px'
];
}
// Make changes to the view only if the resizer should actually get new dimensions.
// Otherwise, if View#change() was always called, this would cause EditorUI#update
// loops because the WidgetResize plugin listens to EditorUI#update and updates
// the resizer.
// https://github.com/ckeditor/ckeditor5/issues/7633
if ( compareArrays( currentDimensions, newDimensions ) !== 'same' ) {
this._options.editor.editing.view.change( writer => {
writer.setStyle( {
width: newDimensions[ 0 ],
height: newDimensions[ 1 ],
left: newDimensions[ 2 ],
top: newDimensions[ 3 ]
}, resizerWrapper );
} );
}
}
containsHandle( domElement ) {
return this._domResizerWrapper.contains( domElement );
}
static isResizeHandle( domElement ) {
return domElement.classList.contains( 'ck-widget__resizer__handle' );
}
/**
* Cleans up the context state.
*
* @protected
*/
_cleanup() {
this._sizeView._dismiss();
const editingView = this._options.editor.editing.view;
editingView.change( writer => {
writer.setStyle( 'width', this._initialViewWidth, this._options.viewElement );
} );
}
/**
* Calculates the proposed size as the resize handles are dragged.
*
* @private
* @param {Event} domEventData Event data that caused the size update request. It should be used to calculate the proposed size.
* @returns {Object} return
* @returns {Number} return.width Proposed width.
* @returns {Number} return.height Proposed height.
*/
_proposeNewSize( domEventData ) {
const state = this.state;
const currentCoordinates = extractCoordinates( domEventData );
const isCentered = this._options.isCentered ? this._options.isCentered( this ) : true;
// Enlargement defines how much the resize host has changed in a given axis. Naturally it could be a negative number
// meaning that it has been shrunk.
//
// +----------------+--+
// | | |
// | img | |
// | /handle host | |
// +----------------+ | ^
// | | | - enlarge y
// +-------------------+ v
// <-->
// enlarge x
const enlargement = {
x: state._referenceCoordinates.x - ( currentCoordinates.x + state.originalWidth ),
y: ( currentCoordinates.y - state.originalHeight ) - state._referenceCoordinates.y
};
if ( isCentered && state.activeHandlePosition.endsWith( '-right' ) ) {
enlargement.x = currentCoordinates.x - ( state._referenceCoordinates.x + state.originalWidth );
}
// Objects needs to be resized twice as much in horizontal axis if centered, since enlargement is counted from
// one resized corner to your cursor. It needs to be duplicated to compensate for the other side too.
if ( isCentered ) {
enlargement.x *= 2;
}
// const resizeHost = this._getResizeHost();
// The size proposed by the user. It does not consider the aspect ratio.
const proposedSize = {
width: Math.abs( state.originalWidth + enlargement.x ),
height: Math.abs( state.originalHeight + enlargement.y )
};
// Dominant determination must take the ratio into account.
proposedSize.dominant = proposedSize.width / state.aspectRatio > proposedSize.height ? 'width' : 'height';
proposedSize.max = proposedSize[ proposedSize.dominant ];
// Proposed size, respecting the aspect ratio.
const targetSize = {
width: proposedSize.width,
height: proposedSize.height
};
if ( proposedSize.dominant == 'width' ) {
targetSize.height = targetSize.width / state.aspectRatio;
} else {
targetSize.width = targetSize.height * state.aspectRatio;
}
return {
width: Math.round( targetSize.width ),
height: Math.round( targetSize.height ),
widthPercents: Math.min( Math.round( state.originalWidthPercents / state.originalWidth * targetSize.width * 100 ) / 100, 100 )
};
}
/**
* Obtains the resize host.
*
* Resize host is an object that receives dimensions which are the result of resizing.
*
* @protected
* @returns {HTMLElement}
*/
_getResizeHost() {
const widgetWrapper = this._domResizerWrapper.parentElement;
return this._options.getResizeHost( widgetWrapper );
}
/**
* Obtains the handle host.
*
* Handle host is an object that the handles are aligned to.
*
* Handle host will not always be an entire widget itself. Take an image as an example. The image widget
* contains an image and a caption. Only the image should be surrounded with handles.
*
* @protected
* @returns {HTMLElement}
*/
_getHandleHost() {
const widgetWrapper = this._domResizerWrapper.parentElement;
return this._options.getHandleHost( widgetWrapper );
}
/**
* DOM container of the entire resize UI.
*
* Note that this property will have a value only after the element bound with the resizer is rendered
* (otherwise `null`).
*
* @private
* @member {HTMLElement|null}
*/
get _domResizerWrapper() {
return this._options.editor.editing.view.domConverter.mapViewToDom( this._viewResizerWrapper );
}
/**
* Renders the resize handles in the DOM.
*
* @private
* @param {HTMLElement} domElement The resizer wrapper.
*/
_appendHandles( domElement ) {
const resizerPositions = [ 'top-left', 'top-right', 'bottom-right', 'bottom-left' ];
for ( const currentPosition of resizerPositions ) {
domElement.appendChild( ( new Template( {
tag: 'div',
attributes: {
class: `ck-widget__resizer__handle ${ getResizerClass( currentPosition ) }`
}
} ).render() ) );
}
}
/**
* Sets up the {@link #_sizeView} property and adds it to the passed `domElement`.
*
* @private
* @param {HTMLElement} domElement
*/
_appendSizeUI( domElement ) {
this._sizeView = new SizeView();
// Make sure icon#element is rendered before passing to appendChild().
this._sizeView.render();
domElement.appendChild( this._sizeView.element );
}
/**
* @event begin
*/
/**
* @event updateSize
*/
/**
* @event commit
*/
/**
* @event cancel
*/
}
mix( Resizer, ObservableMixin );
// @private
// @param {String} resizerPosition Expected resizer position like `"top-left"`, `"bottom-right"`.
// @returns {String} A prefixed HTML class name for the resizer element
function getResizerClass( resizerPosition ) {
return `ck-widget__resizer__handle-${ resizerPosition }`;
}
function extractCoordinates( event ) {
return {
x: event.pageX,
y: event.pageY
};
}
function existsInDom( element ) {
return element && element.ownerDocument && element.ownerDocument.contains( element );
}