%PDF- %PDF-
| Direktori : /www/varak.net/wiki.varak.net/extensions/VisualEditor/lib/ve/src/ce/ |
| Current File : //www/varak.net/wiki.varak.net/extensions/VisualEditor/lib/ve/src/ce/ve.ce.FocusableNode.js |
/*!
* VisualEditor ContentEditable FocusableNode class.
*
* @copyright 2011-2016 VisualEditor Team and others; see http://ve.mit-license.org
*/
/**
* ContentEditable focusable node.
*
* Focusable elements have a special treatment by ve.ce.Surface. When the user selects only a single
* node, if it is focusable, the surface will set the focusable node's focused state. Other systems,
* such as the context, may also use a focusable node's $focusable property as a hint of where the
* primary element in the node is. Typically, and by default, the primary element is the root
* element, but in some cases it may need to be configured to be a specific child element within the
* node's DOM rendering.
*
* If your focusable node changes size and the highlight must be redrawn, call redrawHighlights().
* 'resizeEnd' and 'rerender' are already bound to call this.
*
* @class
* @abstract
*
* @constructor
* @param {jQuery} [$focusable=this.$element] Primary element user is focusing on
* @param {Object} [config] Configuration options
* @cfg {string[]} [classes] CSS classes to be added to the highlight container
*/
ve.ce.FocusableNode = function VeCeFocusableNode( $focusable, config ) {
config = config || {};
// Properties
this.focused = false;
this.highlighted = false;
this.isFocusableSetup = false;
this.$highlights = $( '<div>' ).addClass( 've-ce-focusableNode-highlights' );
this.$focusable = $focusable || this.$element;
this.focusableSurface = null;
this.rects = null;
this.boundingRect = null;
this.startAndEndRects = null;
this.$icon = null;
this.touchMoved = false;
if ( Array.isArray( config.classes ) ) {
this.$highlights.addClass( config.classes.join( ' ' ) );
}
// DOM changes
this.$element
.addClass( 've-ce-focusableNode' )
.prop( 'contentEditable', 'false' );
// Events
this.connect( this, {
setup: 'onFocusableSetup',
teardown: 'onFocusableTeardown',
resizeStart: 'onFocusableResizeStart',
resizeEnd: 'onFocusableResizeEnd',
rerender: 'onFocusableRerender'
} );
};
/* Inheritance */
OO.initClass( ve.ce.FocusableNode );
/* Events */
/**
* @event focus
*/
/**
* @event blur
*/
/* Static properties */
/**
* Icon to use when the rendering is considered not visible, as defined in #hasRendering
*
* No icon is show if null.
*
* @static
* @property {string|null}
* @inheritable
*/
ve.ce.FocusableNode.static.iconWhenInvisible = null;
/* Methods */
/**
* Create a highlight element.
*
* @return {jQuery} A highlight element
*/
ve.ce.FocusableNode.prototype.createHighlight = function () {
var extraClasses = this.generatedContentsInvalid ? ' ve-ce-focusableNode-highlight-error' : '';
return $( '<div>' )
.addClass( 've-ce-focusableNode-highlight' + extraClasses )
.prop( {
title: this.constructor.static.getDescription( this.model ),
draggable: false
} )
.append( $( '<img>' )
.addClass( 've-ce-focusableNode-highlight-relocatable-marker' )
.attr( 'src', '' )
.on( {
mousedown: this.onFocusableMouseDown.bind( this ),
dragstart: this.onFocusableDragStart.bind( this ),
dragend: this.onFocusableDragEnd.bind( this )
} )
);
};
/**
* Handle node setup.
*
* @method
*/
ve.ce.FocusableNode.prototype.onFocusableSetup = function () {
var rAF = requestAnimationFrame || setTimeout;
// Exit if already setup or not attached
if ( this.isFocusableSetup || !this.root ) {
return;
}
this.focusableSurface = this.root.getSurface();
// DOM changes (duplicated from constructor in case this.$element is replaced)
this.$element
.addClass( 've-ce-focusableNode' )
.prop( 'contentEditable', 'false' );
// Events
this.$focusable.on( {
'mouseenter.ve-ce-focusableNode': this.onFocusableMouseEnter.bind( this ),
'touchstart.ve-ce-focusableNode': this.onFocusableTouchStart.bind( this ),
'touchmove.ve-ce-focusableNode': this.onFocusableTouchMove.bind( this ),
'mousedown.ve-ce-focusableNode touchend.ve-ce-focusableNode': this.onFocusableMouseDown.bind( this )
} );
// $element is ce=false so make sure nothing happens when you click
// on it, just in case the browser decides to do something.
// If $element == $focusable then this can be skipped as $focusable already
// handles mousedown events.
if ( !this.$element.is( this.$focusable ) ) {
this.$element.on( {
'mousedown.ve-ce-focusableNode': function ( e ) { e.preventDefault(); }
} );
}
if ( this.constructor.static.iconWhenInvisible ) {
// Set up the invisible icon, and watch for its continued necessity if
// unloaded images which don't specify their width or height are
// involved.
this.$element
.find( 'img:not([width]),img:not([height])' )
.addBack( 'img:not([width]),img:not([height])' )
.on( 'load', this.updateInvisibleIcon.bind( this ) );
rAF( this.updateInvisibleIcon.bind( this ) );
}
this.isFocusableSetup = true;
};
/**
* Update the state of icon if this node is invisible
*
* If the node doesn't have a visible rendering, we insert an icon to represent
* it. If the icon was already present, and this is called again when rendering
* has developed, we remove the icon.
*
* @method
*/
ve.ce.FocusableNode.prototype.updateInvisibleIcon = function () {
if ( !this.constructor.static.iconWhenInvisible ) {
return;
}
if ( !this.hasRendering() ) {
if ( !this.$icon ) {
this.$icon = this.createInvisibleIcon();
}
this.$element.first()
.addClass( 've-ce-focusableNode-invisible' )
.prepend( this.$icon );
} else if ( this.$icon ) {
this.$element.first().removeClass( 've-ce-focusableNode-invisible' );
this.$icon.detach();
}
};
/**
* Create a element to show if the node is invisible
*
* @return {jQuery} Element to show
*/
ve.ce.FocusableNode.prototype.createInvisibleIcon = function () {
var icon = new OO.ui.IconWidget( {
classes: [ 've-ce-focusableNode-invisibleIcon' ],
icon: this.constructor.static.iconWhenInvisible
} );
// Add em space for selection highlighting
icon.$element.text( '\u2003' );
return icon.$element;
};
/**
* Handle node teardown.
*
* @method
*/
ve.ce.FocusableNode.prototype.onFocusableTeardown = function () {
// Exit if not setup or not attached
if ( !this.isFocusableSetup || !this.root ) {
return;
}
// Events
this.$focusable.off( '.ve-ce-focusableNode' );
this.$element.off( '.ve-ce-focusableNode' );
// Highlights
this.clearHighlights();
// DOM changes
this.$element
.removeClass( 've-ce-focusableNode' )
.removeProp( 'contentEditable' );
this.focusableSurface = null;
this.isFocusableSetup = false;
};
/**
* Handle highlight mouse down events.
*
* @method
* @param {jQuery.Event} e Mouse down event
*/
ve.ce.FocusableNode.prototype.onFocusableMouseDown = function ( e ) {
var range,
node = this,
surfaceModel = this.focusableSurface.getModel(),
selection = surfaceModel.getSelection(),
nodeRange = this.model.getOuterRange();
if ( e.type === 'touchend' && this.touchMoved ) {
return;
}
if ( !this.isInContentEditable() ) {
return;
}
if ( e.which === OO.ui.MouseButtons.RIGHT ) {
// Hide images, and select spans so context menu shows 'copy', but not 'copy image'
this.$highlights.addClass( 've-ce-focusableNode-highlights-contextOpen' );
// Make ce=true so we get cut/paste options in context menu
this.$highlights.prop( 'contentEditable', 'true' );
ve.selectElement( this.$highlights[ 0 ] );
setTimeout( function () {
// Undo everything as soon as the context menu is show
node.$highlights.removeClass( 've-ce-focusableNode-highlights-contextOpen' );
node.$highlights.prop( 'contentEditable', 'true' );
node.focusableSurface.preparePasteTargetForCopy();
} );
}
// Wait for native selection to change before correcting
setTimeout( function () {
range = selection instanceof ve.dm.LinearSelection && selection.getRange();
surfaceModel.getLinearFragment(
e.shiftKey && range ?
ve.Range.static.newCoveringRange(
[ range, nodeRange ], range.from > nodeRange.from
) :
nodeRange
).select();
node.focusableSurface.updateActiveLink();
} );
};
/**
* Handle highlight double click events.
*
* @method
* @param {jQuery.Event} e Double click event
*/
ve.ce.FocusableNode.prototype.onFocusableDblClick = function () {
if ( !this.isInContentEditable() ) {
return;
}
this.executeCommand();
};
/**
* Execute the command associated with this node.
*
* @method
*/
ve.ce.FocusableNode.prototype.executeCommand = function () {
var command, surface;
if ( !this.model.isInspectable() ) {
return false;
}
surface = this.focusableSurface.getSurface();
command = surface.commandRegistry.getCommandForNode( this );
if ( command ) {
command.execute( surface );
}
};
/**
* Handle element drag start.
*
* @method
* @param {jQuery.Event} e Drag start event
*/
ve.ce.FocusableNode.prototype.onFocusableDragStart = function () {
if ( this.focusableSurface ) {
// Allow dragging this node in the surface
this.focusableSurface.startRelocation( this );
}
this.$highlights.addClass( 've-ce-focusableNode-highlights-relocating' );
};
/**
* Handle element drag end.
*
* If a relocation actually takes place the node is destroyed before this events fires.
*
* @method
* @param {jQuery.Event} e Drag end event
*/
ve.ce.FocusableNode.prototype.onFocusableDragEnd = function () {
// endRelocation is usually triggered by onDocumentDrop in the surface, but if it isn't
// trigger it here instead
if ( this.focusableSurface ) {
this.focusableSurface.endRelocation();
}
this.$highlights.removeClass( 've-ce-focusableNode-highlights-relocating' );
};
/**
* Handle mouse enter events.
*
* @method
* @param {jQuery.Event} e Mouse enter event
*/
ve.ce.FocusableNode.prototype.onFocusableMouseEnter = function () {
if ( !this.root.getSurface().dragging && !this.root.getSurface().resizing && this.isInContentEditable() ) {
this.createHighlights();
}
};
/**
* Handle touch start events.
*
* @method
* @param {jQuery.Event} e Touch start event
*/
ve.ce.FocusableNode.prototype.onFocusableTouchStart = function () {
this.touchMoved = false;
};
/**
* Handle touch move events.
*
* @method
* @param {jQuery.Event} e Touch move event
*/
ve.ce.FocusableNode.prototype.onFocusableTouchMove = function () {
this.touchMoved = true;
};
/**
* Handle surface mouse move events.
*
* @method
* @param {jQuery.Event} e Mouse move event
*/
ve.ce.FocusableNode.prototype.onSurfaceMouseMove = function ( e ) {
var $target = $( e.target );
if (
!$target.hasClass( 've-ce-focusableNode-highlight' ) &&
!OO.ui.contains( this.$focusable.toArray(), $target[ 0 ], true )
) {
this.clearHighlights();
}
};
/**
* Handle surface mouse leave events.
*
* @method
* @param {jQuery.Event} e Mouse leave event
*/
ve.ce.FocusableNode.prototype.onSurfaceMouseLeave = function ( e ) {
if ( e.relatedTarget === null ) {
this.clearHighlights();
}
};
/**
* Handle resize start events.
*
* @method
*/
ve.ce.FocusableNode.prototype.onFocusableResizeStart = function () {
this.clearHighlights();
};
/**
* Handle resize end event.
*
* @method
*/
ve.ce.FocusableNode.prototype.onFocusableResizeEnd = function () {
this.redrawHighlights();
};
/**
* Handle rerender event.
*
* @method
*/
ve.ce.FocusableNode.prototype.onFocusableRerender = function () {
if ( this.focused && this.focusableSurface ) {
this.redrawHighlights();
// reposition menu
this.focusableSurface.getSurface().getContext().updateDimensions( true );
}
};
/**
* Check if node is focused.
*
* @method
* @return {boolean} Node is focused
*/
ve.ce.FocusableNode.prototype.isFocused = function () {
return this.focused;
};
/**
* Set the selected state of the node.
*
* @method
* @param {boolean} value Node is focused
* @fires focus
* @fires blur
*/
ve.ce.FocusableNode.prototype.setFocused = function ( value ) {
value = !!value;
if ( this.focused !== value ) {
this.focused = value;
if ( this.focused ) {
this.emit( 'focus' );
this.$element.addClass( 've-ce-focusableNode-focused' );
this.createHighlights();
this.focusableSurface.appendHighlights( this.$highlights, this.focused );
this.focusableSurface.$element.off( '.ve-ce-focusableNode' );
this.focusableSurface.connect( this, { position: 'positionHighlights' } );
} else {
this.emit( 'blur' );
this.$element.removeClass( 've-ce-focusableNode-focused' );
this.clearHighlights();
}
}
};
/**
* Creates highlights.
*
* @method
*/
ve.ce.FocusableNode.prototype.createHighlights = function () {
if ( this.highlighted ) {
return;
}
this.$highlights.on( {
mousedown: this.onFocusableMouseDown.bind( this ),
dblclick: this.onFocusableDblClick.bind( this )
} );
this.highlighted = true;
this.positionHighlights();
this.focusableSurface.appendHighlights( this.$highlights, this.focused );
// Events
if ( !this.focused ) {
this.focusableSurface.$element.on( {
'mousemove.ve-ce-focusableNode': this.onSurfaceMouseMove.bind( this ),
'mouseleave.ve-ce-focusableNode': this.onSurfaceMouseLeave.bind( this )
} );
}
};
/**
* Clears highlight.
*
* @method
*/
ve.ce.FocusableNode.prototype.clearHighlights = function () {
if ( !this.highlighted ) {
return;
}
this.$highlights.remove().empty();
this.focusableSurface.$element.off( '.ve-ce-focusableNode' );
this.focusableSurface.disconnect( this, { position: 'positionHighlights' } );
this.highlighted = false;
this.boundingRect = null;
};
/**
* Redraws highlight.
*
* @method
*/
ve.ce.FocusableNode.prototype.redrawHighlights = function () {
this.clearHighlights();
this.createHighlights();
};
/**
* Calculate position of highlights
*/
ve.ce.FocusableNode.prototype.calculateHighlights = function () {
var i, l, $set, columnCount, columnWidth, surfaceOffset,
rects = [],
filteredRects = [],
webkitColumns = 'webkitColumnCount' in document.createElement( 'div' ).style;
// Protect against calling before/after surface setup/teardown
if ( !this.focusableSurface ) {
this.boundingRect = null;
this.startAndEndRects = null;
this.rects = [];
return;
}
surfaceOffset = this.focusableSurface.getSurface().getBoundingClientRect();
function contains( rect1, rect2 ) {
return rect2.left >= rect1.left &&
rect2.top >= rect1.top &&
rect2.right <= rect1.right &&
rect2.bottom <= rect1.bottom;
}
function process( el ) {
var i, j, il, jl, contained, clientRects, overflow, $el;
if ( el.classList.contains( 've-ce-noHighlight' ) ) {
return;
}
$el = $( el );
if ( webkitColumns ) {
columnCount = $el.css( '-webkit-column-count' );
columnWidth = $el.css( '-webkit-column-width' );
if ( ( columnCount && columnCount !== 'auto' ) || ( columnWidth && columnWidth !== 'auto' ) ) {
// Support: Chrome
// Chrome incorrectly measures children of nodes with columns [1], let's
// just ignore them rather than render a possibly bizarre highlight. They
// will usually not be positioned, because Chrome also doesn't position
// them correctly [2] and so people avoid doing it.
//
// Of course there are other ways to render a node outside the bounding
// box of its parent, like negative margin. We do not handle these cases,
// and the highlight may not correctly cover the entire node if that
// happens. This can't be worked around without implementing CSS
// layouting logic ourselves, which is not worth it.
//
// [1] https://code.google.com/p/chromium/issues/detail?id=391271
// [2] https://code.google.com/p/chromium/issues/detail?id=291616
// jQuery keeps nodes in its collections in document order, so the
// children have not been processed yet and can be safely removed.
$set = $set.not( $el.find( '*' ) );
}
}
// Don't descend if overflow is anything but visible as this prevents
// child elements appearing beyond the bounding box of the parent
overflow = $el.css( 'overflow' );
if ( overflow && overflow !== 'visible' ) {
$set = $set.not( $el.find( '*' ) );
}
clientRects = el.getClientRects();
for ( i = 0, il = clientRects.length; i < il; i++ ) {
contained = false;
for ( j = 0, jl = rects.length; j < jl; j++ ) {
// This rect is contained by an existing rect, discard
if ( contains( rects[ j ], clientRects[ i ] ) ) {
contained = true;
break;
}
// An existing rect is contained by this rect, discard the existing rect
if ( contains( clientRects[ i ], rects[ j ] ) ) {
rects.splice( j, 1 );
j--;
jl--;
}
}
if ( !contained ) {
rects.push( clientRects[ i ] );
}
}
}
$set = this.$focusable.find( '*' ).addBack();
// Calling process() may change $set.length
for ( i = 0; i < $set.length; i++ ) {
process( $set[ i ] );
}
// Elements with a width/height of 0 return a clientRect with a width/height of 1
// As elements with an actual width/height of 1 aren't that useful anyway, just
// throw away anything that is <=1
filteredRects = rects.filter( function ( rect ) {
return rect.width > 1 && rect.height > 1;
} );
// But if this filtering doesn't leave any rects at all, then we do want to use the 1px rects
if ( filteredRects.length > 0 ) {
rects = filteredRects;
}
this.boundingRect = null;
// startAndEndRects is lazily evaluated in getStartAndEndRects from rects
this.startAndEndRects = null;
for ( i = 0, l = rects.length; i < l; i++ ) {
// Translate to relative
rects[ i ] = ve.translateRect( rects[ i ], -surfaceOffset.left, -surfaceOffset.top );
this.$highlights.append(
this.createHighlight().css( {
top: rects[ i ].top,
left: rects[ i ].left,
width: rects[ i ].width,
height: rects[ i ].height
} )
);
if ( !this.boundingRect ) {
this.boundingRect = ve.copy( rects[ i ] );
} else {
this.boundingRect.top = Math.min( this.boundingRect.top, rects[ i ].top );
this.boundingRect.left = Math.min( this.boundingRect.left, rects[ i ].left );
this.boundingRect.bottom = Math.max( this.boundingRect.bottom, rects[ i ].bottom );
this.boundingRect.right = Math.max( this.boundingRect.right, rects[ i ].right );
}
}
if ( this.boundingRect ) {
this.boundingRect.width = this.boundingRect.right - this.boundingRect.left;
this.boundingRect.height = this.boundingRect.bottom - this.boundingRect.top;
}
this.rects = rects;
};
/**
* Positions highlights, and remove collapsed ones
*
* @method
*/
ve.ce.FocusableNode.prototype.positionHighlights = function () {
var i, l;
if ( !this.highlighted ) {
return;
}
this.calculateHighlights();
this.$highlights.empty()
// Append something selectable for right-click copy
.append( $( '<span>' ).addClass( 've-ce-focusableNode-highlight-selectable' ).html( ' ' ) );
for ( i = 0, l = this.rects.length; i < l; i++ ) {
this.$highlights.append(
this.createHighlight().css( {
top: this.rects[ i ].top,
left: this.rects[ i ].left,
width: this.rects[ i ].width,
height: this.rects[ i ].height
} )
);
}
};
/**
* Get list of rectangles outlining the shape of the node relative to the surface
*
* @return {Object[]} List of rectangle objects
*/
ve.ce.FocusableNode.prototype.getRects = function () {
if ( !this.highlighted ) {
this.calculateHighlights();
}
return this.rects;
};
/**
* Get the bounding rectangle of the focusable node highlight relative to the surface
*
* @return {Object|null} Top, left, bottom & right positions of the focusable node relative to the surface
*/
ve.ce.FocusableNode.prototype.getBoundingRect = function () {
if ( !this.highlighted ) {
this.calculateHighlights();
}
return this.boundingRect;
};
/**
* Get start and end rectangles of an inline focusable node relative to the surface
*
* @return {Object|null} Start and end rectangles
*/
ve.ce.FocusableNode.prototype.getStartAndEndRects = function () {
if ( !this.highlighted ) {
this.calculateHighlights();
}
if ( !this.startAndEndRects ) {
this.startAndEndRects = ve.getStartAndEndRects( this.rects );
}
return this.startAndEndRects;
};
/**
* Check if the rendering is visible
*
* "Visible", in this case, is defined as any of:
* * contains any non-whitespace text
* * is greater than 8px x 8px in dimensions
*
* @return {boolean} The node has a visible rendering
*/
ve.ce.FocusableNode.prototype.hasRendering = function () {
var visible = false;
if ( this.$element.text().trim() !== '' ) {
return true;
}
this.$element.each( function () {
var $this = $( this );
if (
( $this.width() >= 8 && $this.height() >= 8 ) ||
// jQuery handles disparate cases, but is prone to elements which
// haven't experienced layout yet having 0 width / height. So,
// check the raw DOM width / height properties as well. If it's an
// image or other thing-with-width, this will work slightly more
// reliably. If it's not, this will be undefined and the
// comparison will thus just be false.
( this.width >= 8 && this.height >= 8 )
) {
visible = true;
return false;
}
} );
return visible;
};