2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* This plugin provides a way to build a wiki-text editing user interface around a textarea.
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* @example To intialize without any modules:
|
2012-07-03 05:52:27 +00:00
|
|
|
* $( 'div#edittoolbar' ).wikiEditor();
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* @example To initialize with one or more modules, or to add modules after it's already been initialized:
|
2012-07-03 05:52:27 +00:00
|
|
|
* $( 'textarea#wpTextbox1' ).wikiEditor( 'addModule', 'toolbar', { ... config ... } );
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
*/
|
2013-11-07 00:38:43 +00:00
|
|
|
/*jshint onevar:false, boss:true */
|
|
|
|
( function ( $, mw ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
|
2015-01-13 12:33:47 +00:00
|
|
|
var hasOwn = Object.prototype.hasOwnProperty;
|
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Global static object for wikiEditor that provides generally useful functionality to all modules and contexts.
|
|
|
|
*/
|
|
|
|
$.wikiEditor = {
|
|
|
|
/**
|
|
|
|
* For each module that is loaded, static code shared by all instances is loaded into this object organized by
|
|
|
|
* module name. The existance of a module in this object only indicates the module is available. To check if a
|
|
|
|
* module is in use by a specific context check the context.modules object.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
modules: {},
|
|
|
|
|
2010-10-04 20:12:30 +00:00
|
|
|
/**
|
|
|
|
* A context can be extended, such as adding iframe support, on a per-wikiEditor instance basis.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
extensions: {},
|
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* In some cases like with the iframe's HTML file, it's convienent to have a lookup table of all instances of the
|
|
|
|
* WikiEditor. Each context contains an instance field which contains a key that corrosponds to a reference to the
|
|
|
|
* textarea which the WikiEditor was build around. This way, by passing a simple integer you can provide a way back
|
|
|
|
* to a specific context.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
instances: [],
|
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* For each browser name, an array of conditions that must be met are supplied in [operaton, value]-form where
|
|
|
|
* operation is a string containing a JavaScript compatible binary operator and value is either a number to be
|
|
|
|
* compared with $.browser.versionNumber or a string to be compared with $.browser.version. If a browser is not
|
|
|
|
* specifically mentioned, we just assume things will work.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
browsers: {
|
2010-09-15 03:08:35 +00:00
|
|
|
// Left-to-right languages
|
2012-07-03 05:52:27 +00:00
|
|
|
ltr: {
|
2010-09-15 03:08:35 +00:00
|
|
|
// The toolbar layout is broken in IE6
|
2012-07-03 05:52:27 +00:00
|
|
|
msie: [['>=', 7]],
|
2010-09-15 03:08:35 +00:00
|
|
|
// Layout issues in FF < 2
|
2012-07-03 05:52:27 +00:00
|
|
|
firefox: [['>=', 2]],
|
2014-04-03 14:59:58 +00:00
|
|
|
// Text selection bugs galore
|
2012-07-03 05:52:27 +00:00
|
|
|
opera: [['>=', 9.6]],
|
2010-09-15 03:08:35 +00:00
|
|
|
// jQuery minimums
|
2012-07-03 05:52:27 +00:00
|
|
|
safari: [['>=', 3]],
|
|
|
|
chrome: [['>=', 3]],
|
|
|
|
netscape: [['>=', 9]],
|
|
|
|
blackberry: false,
|
2014-05-10 08:12:54 +00:00
|
|
|
ipod: [['>=', 6]],
|
|
|
|
iphone: [['>=', 6]]
|
2010-09-15 03:08:35 +00:00
|
|
|
},
|
|
|
|
// Right-to-left languages
|
2012-07-03 05:52:27 +00:00
|
|
|
rtl: {
|
2010-09-15 03:08:35 +00:00
|
|
|
// The toolbar layout is broken in IE 7 in RTL mode, and IE6 in any mode
|
2012-07-03 05:52:27 +00:00
|
|
|
msie: [['>=', 8]],
|
2010-09-15 03:08:35 +00:00
|
|
|
// Layout issues in FF < 2
|
2012-07-03 05:52:27 +00:00
|
|
|
firefox: [['>=', 2]],
|
2014-04-03 14:59:58 +00:00
|
|
|
// Text selection bugs galore
|
2012-07-03 05:52:27 +00:00
|
|
|
opera: [['>=', 9.6]],
|
2010-09-15 03:08:35 +00:00
|
|
|
// jQuery minimums
|
2012-07-03 05:52:27 +00:00
|
|
|
safari: [['>=', 3]],
|
|
|
|
chrome: [['>=', 3]],
|
|
|
|
netscape: [['>=', 9]],
|
|
|
|
blackberry: false,
|
2014-05-10 08:12:54 +00:00
|
|
|
ipod: [['>=', 6]],
|
|
|
|
iphone: [['>=', 6]]
|
2010-09-15 03:08:35 +00:00
|
|
|
}
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Path to images - this is a bit messy, and it would need to change if this code (and images) gets moved into the
|
|
|
|
* core - or anywhere for that matter...
|
|
|
|
*/
|
2015-01-14 07:14:44 +00:00
|
|
|
imgPath: mw.config.get( 'wgExtensionAssetsPath' ) + '/WikiEditor/modules/images/',
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Checks the current browser against the browsers object to determine if the browser has been black-listed or not.
|
|
|
|
* Because these rules are often very complex, the object contains configurable operators and can check against
|
|
|
|
* either the browser version number or string. This process also involves checking if the current browser is amung
|
|
|
|
* those which we have configured as compatible or not. If the browser was not configured as comptible we just go on
|
|
|
|
* assuming things will work - the argument here is to prevent the need to update the code when a new browser comes
|
|
|
|
* to market. The assumption here is that any new browser will be built on an existing engine or be otherwise so
|
|
|
|
* similar to another existing browser that things actually do work as expected. The merrits of this argument, which
|
|
|
|
* is essentially to blacklist rather than whitelist are debateable, but at this point we've decided it's the more
|
|
|
|
* "open-web" way to go.
|
|
|
|
* @param module Module object, defaults to $.wikiEditor
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
isSupported: function ( module ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
// Fallback to the wikiEditor browser map if no special map is provided in the module
|
|
|
|
var mod = module && 'browsers' in module ? module : $.wikiEditor;
|
|
|
|
// Check for and make use of cached value and early opportunities to bail
|
|
|
|
if ( typeof mod.supported !== 'undefined' ) {
|
|
|
|
// Cache hit
|
|
|
|
return mod.supported;
|
|
|
|
}
|
|
|
|
// Run a browser support test and then cache and return the result
|
2010-09-15 22:40:50 +00:00
|
|
|
return mod.supported = $.client.test( mod.browsers );
|
2010-09-15 03:08:35 +00:00
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Checks if a module has a specific requirement
|
|
|
|
* @param module Module object
|
|
|
|
* @param requirement String identifying requirement
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
isRequired: function ( module, requirement ) {
|
|
|
|
if ( typeof module.req !== 'undefined' ) {
|
|
|
|
for ( var req in module.req ) {
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( module.req[req] === requirement ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
2015-02-15 03:31:27 +00:00
|
|
|
* Provides a way to extract messages from objects. Wraps a mediaWiki.message( ... ).plain() call.
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* @param object Object to extract messages from
|
|
|
|
* @param property String of name of property which contains the message. This should be the base name of the
|
|
|
|
* property, which means that in the case of the object { this: 'that', fooMsg: 'bar' }, passing property as 'this'
|
|
|
|
* would return the raw text 'that', while passing property as 'foo' would return the internationalized message
|
|
|
|
* with the key 'bar'.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
autoMsg: function ( object, property ) {
|
2013-11-07 00:38:43 +00:00
|
|
|
var i, p;
|
2010-09-15 03:08:35 +00:00
|
|
|
// Accept array of possible properties, of which the first one found will be used
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( typeof property === 'object' ) {
|
|
|
|
for ( i in property ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
if ( property[i] in object || property[i] + 'Msg' in object ) {
|
|
|
|
property = property[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( property in object ) {
|
|
|
|
return object[property];
|
|
|
|
} else if ( property + 'Msg' in object ) {
|
2013-11-07 00:38:43 +00:00
|
|
|
p = object[property + 'Msg'];
|
2010-10-26 23:42:23 +00:00
|
|
|
if ( $.isArray( p ) && p.length >= 2 ) {
|
2013-11-07 00:38:43 +00:00
|
|
|
return mw.message.apply( mw.message, p ).plain();
|
2010-09-15 03:08:35 +00:00
|
|
|
} else {
|
2013-11-07 00:38:43 +00:00
|
|
|
return mw.message( p ).plain();
|
2010-09-15 03:08:35 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Provides a way to extract a property of an object in a certain language, falling back on the property keyed as
|
2011-04-25 12:26:20 +00:00
|
|
|
* 'default' or 'default-rtl'. If such key doesn't exist, the object itself is considered the actual value, which
|
|
|
|
* should ideally be the case so that you may use a string or object of any number of strings keyed by language
|
|
|
|
* with a default.
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* @param object Object to extract property from
|
|
|
|
* @param lang Language code, defaults to wgUserLanguage
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
autoLang: function ( object, lang ) {
|
2011-04-25 12:26:20 +00:00
|
|
|
var defaultKey = $( 'body' ).hasClass( 'rtl' ) ? 'default-rtl' : 'default';
|
2015-01-13 12:33:47 +00:00
|
|
|
lang = lang || mw.config.get( 'wgUserLanguage' );
|
|
|
|
return hasOwn.call( object, lang ) ? object[lang] : ( object[defaultKey] || object['default'] || object );
|
2010-09-15 03:08:35 +00:00
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Provides a way to extract the path of an icon in a certain language, automatically appending a version number for
|
|
|
|
* caching purposes and prepending an image path when icon paths are relative.
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* @param icon Icon object from e.g. toolbar config
|
|
|
|
* @param path Default icon path, defaults to $.wikiEditor.imgPath
|
|
|
|
* @param lang Language code, defaults to wgUserLanguage
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
autoIcon: function ( icon, path, lang ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
var src = $.wikiEditor.autoLang( icon, lang );
|
|
|
|
path = path || $.wikiEditor.imgPath;
|
|
|
|
// Prepend path if src is not absolute
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( src.substr( 0, 7 ) !== 'http://' && src.substr( 0, 8 ) !== 'https://' && src[0] !== '/' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
src = path + src;
|
|
|
|
}
|
2014-02-18 22:53:52 +00:00
|
|
|
return src + '?' + mw.loader.getVersion( 'jquery.wikiEditor' );
|
2010-09-15 03:08:35 +00:00
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Get the sprite offset for a language if available, icon for a language if available, or the default offset or icon,
|
|
|
|
* in that order of preference.
|
|
|
|
* @param icon Icon object, see autoIcon()
|
|
|
|
* @param offset Offset object
|
|
|
|
* @param path Icon path, see autoIcon()
|
|
|
|
* @param lang Language code, defaults to wgUserLanguage
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
autoIconOrOffset: function ( icon, offset, path, lang ) {
|
2011-06-26 19:25:27 +00:00
|
|
|
lang = lang || mw.config.get( 'wgUserLanguage' );
|
2015-01-13 12:33:47 +00:00
|
|
|
if ( typeof offset === 'object' && hasOwn.call( offset, lang ) ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
return offset[lang];
|
2015-01-13 12:33:47 +00:00
|
|
|
} else if ( typeof icon === 'object' && hasOwn.call( icon, lang ) ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
return $.wikiEditor.autoIcon( icon, undefined, lang );
|
|
|
|
} else {
|
|
|
|
return $.wikiEditor.autoLang( offset, lang );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* jQuery plugin that provides a way to initialize a wikiEditor instance on a textarea.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
$.fn.wikiEditor = function () {
|
2010-09-15 03:08:35 +00:00
|
|
|
|
|
|
|
// Skip any further work when running in browsers that are unsupported
|
2011-06-06 23:18:13 +00:00
|
|
|
if ( !$.wikiEditor.isSupported() ) {
|
2014-04-30 10:59:01 +00:00
|
|
|
return $( this );
|
2010-09-15 03:08:35 +00:00
|
|
|
}
|
|
|
|
|
2014-12-06 12:40:50 +00:00
|
|
|
// Save browser profile for detailed tests.
|
|
|
|
var profile = $.client.profile();
|
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/* Initialization */
|
|
|
|
|
|
|
|
// The wikiEditor context is stored in the element's data, so when this function gets called again we can pick up right
|
|
|
|
// where we left off
|
2014-04-30 10:59:01 +00:00
|
|
|
var context = $( this ).data( 'wikiEditor-context' );
|
2010-09-15 03:08:35 +00:00
|
|
|
// On first call, we need to set things up, but on all following calls we can skip right to the API handling
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( !context || typeof context === 'undefined' ) {
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
// Star filling the context with useful data - any jQuery selections, as usual should be named with a preceding $
|
|
|
|
context = {
|
|
|
|
// Reference to the textarea element which the wikiEditor is being built around
|
2014-04-30 10:59:01 +00:00
|
|
|
'$textarea': $( this ),
|
2010-09-15 03:08:35 +00:00
|
|
|
// Container for any number of mutually exclusive views that are accessible by tabs
|
|
|
|
'views': {},
|
|
|
|
// Container for any number of module-specific data - only including data for modules in use on this context
|
|
|
|
'modules': {},
|
|
|
|
// General place to shouve bits of data into
|
|
|
|
'data': {},
|
|
|
|
// Unique numeric ID of this instance used both for looking up and differentiating instances of wikiEditor
|
2014-04-30 10:59:01 +00:00
|
|
|
'instance': $.wikiEditor.instances.push( $( this ) ) - 1,
|
2014-01-05 16:49:54 +00:00
|
|
|
// Saved selection state for old IE (<=10)
|
2010-09-15 03:08:35 +00:00
|
|
|
'savedSelection': null,
|
2010-10-04 20:12:30 +00:00
|
|
|
// List of extensions active on this context
|
2010-12-06 11:07:50 +00:00
|
|
|
'extensions': []
|
2010-09-15 03:08:35 +00:00
|
|
|
};
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2011-10-26 03:49:06 +00:00
|
|
|
/**
|
2010-09-15 03:08:35 +00:00
|
|
|
* Externally Accessible API
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2014-04-30 10:59:01 +00:00
|
|
|
* These are available using calls to $( selection ).wikiEditor( call, data ) where selection is a jQuery selection
|
2010-09-15 03:08:35 +00:00
|
|
|
* of the textarea that the wikiEditor instance was built around.
|
|
|
|
*/
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
context.api = {
|
|
|
|
/**
|
|
|
|
* Activates a module on a specific context with optional configuration data.
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* @param data Either a string of the name of a module to add without any additional configuration parameters,
|
|
|
|
* or an object with members keyed with module names and valued with configuration objects.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
'addModule': function ( context, data ) {
|
|
|
|
var module, call,
|
|
|
|
modules = {};
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( typeof data === 'string' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
modules[data] = {};
|
2013-11-07 00:38:43 +00:00
|
|
|
} else if ( typeof data === 'object' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
modules = data;
|
|
|
|
}
|
2012-07-03 05:52:27 +00:00
|
|
|
for ( module in modules ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
// Check for the existance of an available / supported module with a matching name and a create function
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( typeof module === 'string' && typeof $.wikiEditor.modules[module] !== 'undefined' &&
|
2011-02-23 20:50:52 +00:00
|
|
|
$.wikiEditor.isSupported( $.wikiEditor.modules[module] ) )
|
|
|
|
{
|
2010-09-15 03:08:35 +00:00
|
|
|
// Extend the context's core API with this module's own API calls
|
|
|
|
if ( 'api' in $.wikiEditor.modules[module] ) {
|
2012-07-03 05:52:27 +00:00
|
|
|
for ( call in $.wikiEditor.modules[module].api ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
// Modules may not overwrite existing API functions - first come, first serve
|
|
|
|
if ( !( call in context.api ) ) {
|
|
|
|
context.api[call] = $.wikiEditor.modules[module].api[call];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Activate the module on this context
|
|
|
|
if ( 'fn' in $.wikiEditor.modules[module] && 'create' in $.wikiEditor.modules[module].fn ) {
|
|
|
|
// Add a place for the module to put it's own stuff
|
|
|
|
context.modules[module] = {};
|
|
|
|
// Tell the module to create itself on the context
|
|
|
|
$.wikiEditor.modules[module].fn.create( context, modules[module] );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2011-10-26 03:49:06 +00:00
|
|
|
/**
|
2010-09-15 03:08:35 +00:00
|
|
|
* Event Handlers
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* These act as filters returning false if the event should be ignored or returning true if it should be passed
|
|
|
|
* on to all modules. This is also where we can attach some extra information to the events.
|
|
|
|
*/
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
context.evt = {
|
2011-06-09 18:10:02 +00:00
|
|
|
/* Empty until extensions add some; see jquery.wikiEditor.iframe.js for examples. */
|
2010-09-15 03:08:35 +00:00
|
|
|
};
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/* Internal Functions */
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
context.fn = {
|
|
|
|
/**
|
|
|
|
* Executes core event filters as well as event handlers provided by modules.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
trigger: function ( name, event ) {
|
2014-12-06 12:40:50 +00:00
|
|
|
// Workaround for a scrolling bug in IE8 (bug 61908)
|
|
|
|
if ( profile.name === 'msie' && profile.versionNumber === 8 ) {
|
|
|
|
context.$textarea.css( 'width', context.$textarea.parent().width() );
|
|
|
|
}
|
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
// Event is an optional argument, but from here on out, at least the type field should be dependable
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( typeof event === 'undefined' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
event = { 'type': 'custom' };
|
|
|
|
}
|
|
|
|
// Ensure there's a place for extra information to live
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( typeof event.data === 'undefined' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
event.data = {};
|
|
|
|
}
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
// Allow filtering to occur
|
|
|
|
if ( name in context.evt ) {
|
|
|
|
if ( !context.evt[name]( event ) ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2015-03-14 15:49:04 +00:00
|
|
|
var returnFromModules = null; // they return null by default
|
2010-09-15 03:08:35 +00:00
|
|
|
// Pass the event around to all modules activated on this context
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
for ( var module in context.modules ) {
|
|
|
|
if (
|
|
|
|
module in $.wikiEditor.modules &&
|
|
|
|
'evt' in $.wikiEditor.modules[module] &&
|
|
|
|
name in $.wikiEditor.modules[module].evt
|
|
|
|
) {
|
|
|
|
var ret = $.wikiEditor.modules[module].evt[name]( context, event );
|
2014-04-30 10:59:01 +00:00
|
|
|
if ( ret !== null ) {
|
2015-03-14 15:49:04 +00:00
|
|
|
// if 1 returns false, the end result is false
|
2014-04-30 10:59:01 +00:00
|
|
|
if ( returnFromModules === null ) {
|
2011-09-13 08:56:32 +00:00
|
|
|
returnFromModules = ret;
|
2010-09-15 03:08:35 +00:00
|
|
|
} else {
|
|
|
|
returnFromModules = returnFromModules && ret;
|
2011-09-13 08:56:32 +00:00
|
|
|
}
|
2010-09-15 03:08:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-07-03 05:52:27 +00:00
|
|
|
if ( returnFromModules !== null ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
return returnFromModules;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Adds a button to the UI
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
addButton: function ( options ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
// Ensure that buttons and tabs are visible
|
|
|
|
context.$controls.show();
|
|
|
|
context.$buttons.show();
|
2012-07-03 05:52:27 +00:00
|
|
|
return $( '<button>' )
|
2010-09-15 03:08:35 +00:00
|
|
|
.text( $.wikiEditor.autoMsg( options, 'caption' ) )
|
|
|
|
.click( options.action )
|
|
|
|
.appendTo( context.$buttons );
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
|
|
|
* Adds a view to the UI, which is accessed using a set of tabs. Views are mutually exclusive and by default a
|
|
|
|
* wikitext view will be present. Only when more than one view exists will the tabs will be visible.
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
addView: function ( options ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
// Adds a tab
|
|
|
|
function addTab( options ) {
|
|
|
|
// Ensure that buttons and tabs are visible
|
|
|
|
context.$controls.show();
|
|
|
|
context.$tabs.show();
|
|
|
|
// Return the newly appended tab
|
2012-07-03 05:52:27 +00:00
|
|
|
return $( '<div>' )
|
2010-09-15 03:08:35 +00:00
|
|
|
.attr( 'rel', 'wikiEditor-ui-view-' + options.name )
|
2013-11-07 00:38:43 +00:00
|
|
|
.addClass( context.view === options.name ? 'current' : null )
|
2012-07-03 05:52:27 +00:00
|
|
|
.append( $( '<a>' )
|
2010-09-15 03:08:35 +00:00
|
|
|
.attr( 'href', '#' )
|
2012-07-03 05:52:27 +00:00
|
|
|
.mousedown( function () {
|
2010-09-15 03:08:35 +00:00
|
|
|
// No dragging!
|
|
|
|
return false;
|
|
|
|
} )
|
2012-07-03 05:52:27 +00:00
|
|
|
.click( function ( event ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
context.$ui.find( '.wikiEditor-ui-view' ).hide();
|
2014-04-30 10:59:01 +00:00
|
|
|
context.$ui.find( '.' + $( this ).parent().attr( 'rel' ) ).show();
|
2010-09-15 03:08:35 +00:00
|
|
|
context.$tabs.find( 'div' ).removeClass( 'current' );
|
2014-04-30 10:59:01 +00:00
|
|
|
$( this ).parent().addClass( 'current' );
|
|
|
|
$( this ).blur();
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( 'init' in options && typeof options.init === 'function' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
options.init( context );
|
|
|
|
}
|
|
|
|
event.preventDefault();
|
|
|
|
return false;
|
|
|
|
} )
|
|
|
|
.text( $.wikiEditor.autoMsg( options, 'title' ) )
|
|
|
|
)
|
|
|
|
.appendTo( context.$tabs );
|
|
|
|
}
|
|
|
|
// Automatically add the previously not-needed wikitext tab
|
2012-07-03 05:52:27 +00:00
|
|
|
if ( !context.$tabs.children().length ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
addTab( { 'name': 'wikitext', 'titleMsg': 'wikieditor-wikitext-tab' } );
|
|
|
|
}
|
|
|
|
// Add the tab for the view we were actually asked to add
|
|
|
|
addTab( options );
|
|
|
|
// Return newly appended view
|
2012-07-03 05:52:27 +00:00
|
|
|
return $( '<div>' )
|
2010-09-15 03:08:35 +00:00
|
|
|
.addClass( 'wikiEditor-ui-view wikiEditor-ui-view-' + options.name )
|
|
|
|
.hide()
|
|
|
|
.appendTo( context.$ui );
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
2015-02-07 02:14:40 +00:00
|
|
|
* Save scrollTop and cursor position for old IE (<=10)
|
|
|
|
* Related to old IE 8 issues that are no longer reproducible
|
2010-09-15 03:08:35 +00:00
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
saveCursorAndScrollTop: function () {
|
2015-02-07 02:14:40 +00:00
|
|
|
if ( profile.name === 'msie' && document.selection && document.selection.createRange ) {
|
|
|
|
var IHateIE8 = {
|
2015-01-14 07:14:44 +00:00
|
|
|
'scrollTop': context.$textarea.scrollTop(),
|
2010-10-05 20:54:38 +00:00
|
|
|
'pos': context.$textarea.textSelection( 'getCaretPosition', { startAndEnd: true } )
|
|
|
|
};
|
2015-02-07 02:14:40 +00:00
|
|
|
context.$textarea.data( 'IHateIE8', IHateIE8 );
|
2010-10-05 20:54:38 +00:00
|
|
|
}
|
2010-09-15 03:08:35 +00:00
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
/**
|
2015-02-07 02:14:40 +00:00
|
|
|
* Restore scrollTo and cursor position for IE (<=10)
|
|
|
|
* Related to old IE 8 issues that are no longer reproducible
|
2010-09-15 03:08:35 +00:00
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
restoreCursorAndScrollTop: function () {
|
2015-02-07 02:14:40 +00:00
|
|
|
if ( profile.name === 'msie' && document.selection && document.selection.createRange ) {
|
|
|
|
var IHateIE8 = context.$textarea.data( 'IHateIE' );
|
|
|
|
if ( IHateIE8 ) {
|
|
|
|
context.$textarea.scrollTop( IHateIE8.scrollTop );
|
|
|
|
context.$textarea.textSelection( 'setSelection', { start: IHateIE8.pos[0], end: IHateIE8.pos[1] } );
|
|
|
|
context.$textarea.data( 'IHateIE8', null );
|
2010-10-05 20:54:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-10-05 20:54:38 +00:00
|
|
|
/**
|
2014-01-05 16:49:54 +00:00
|
|
|
* Save text selection for old IE (<=10)
|
2010-10-05 20:54:38 +00:00
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
saveSelection: function () {
|
2015-03-02 18:48:42 +00:00
|
|
|
if ( profile.name === 'msie' && document.selection && document.selection.createRange ) {
|
2010-10-05 20:54:38 +00:00
|
|
|
context.$textarea.focus();
|
|
|
|
context.savedSelection = document.selection.createRange();
|
|
|
|
}
|
|
|
|
},
|
2012-07-03 05:52:27 +00:00
|
|
|
|
2010-10-05 20:54:38 +00:00
|
|
|
/**
|
2014-01-05 16:49:54 +00:00
|
|
|
* Restore text selection for old IE (<=10)
|
2010-10-05 20:54:38 +00:00
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
restoreSelection: function () {
|
2015-03-02 18:48:42 +00:00
|
|
|
if ( profile.name === 'msie' && context.savedSelection !== null ) {
|
2010-10-05 20:54:38 +00:00
|
|
|
context.$textarea.focus();
|
|
|
|
context.savedSelection.select();
|
|
|
|
context.savedSelection = null;
|
|
|
|
}
|
2010-12-06 11:07:50 +00:00
|
|
|
}
|
2010-09-15 03:08:35 +00:00
|
|
|
};
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2014-02-25 12:41:12 +00:00
|
|
|
/**
|
|
|
|
* Workaround for a scrolling bug in IE8 (bug 61908)
|
|
|
|
*/
|
2014-12-06 12:40:50 +00:00
|
|
|
if ( profile.name === 'msie' && profile.versionNumber === 8 ) {
|
2014-02-25 12:41:12 +00:00
|
|
|
context.$textarea.css( 'height', context.$textarea.height() );
|
2014-12-06 12:40:50 +00:00
|
|
|
context.$textarea.css( 'width', context.$textarea.parent().width() );
|
2014-02-25 12:41:12 +00:00
|
|
|
}
|
|
|
|
|
2011-10-26 03:49:06 +00:00
|
|
|
/**
|
2010-09-15 03:08:35 +00:00
|
|
|
* Base UI Construction
|
2011-09-13 08:56:32 +00:00
|
|
|
*
|
2010-09-15 03:08:35 +00:00
|
|
|
* The UI is built from several containers, the outer-most being a div classed as "wikiEditor-ui". These containers
|
|
|
|
* provide a certain amount of "free" layout, but in some situations procedural layout is needed, which is performed
|
|
|
|
* as a response to the "resize" event.
|
|
|
|
*/
|
2011-09-13 08:56:32 +00:00
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
// Assemble a temporary div to place over the wikiEditor while it's being constructed
|
|
|
|
/* Disabling our loading div for now
|
2012-07-03 05:52:27 +00:00
|
|
|
var $loader = $( '<div>' )
|
2010-09-15 03:08:35 +00:00
|
|
|
.addClass( 'wikiEditor-ui-loading' )
|
2010-10-27 00:16:32 +00:00
|
|
|
.append( $( '<span>' + mediaWiki.msg( 'wikieditor-loading' ) + '</span>' )
|
2010-09-15 03:08:35 +00:00
|
|
|
.css( 'marginTop', context.$textarea.height() / 2 ) );
|
|
|
|
*/
|
2014-04-29 22:43:32 +00:00
|
|
|
/* Preserving cursor and focus state, which will get lost due to wrapAll */
|
|
|
|
var hasFocus = context.$textarea.is( ':focus' ),
|
2015-03-24 21:14:16 +00:00
|
|
|
cursorPos = context.$textarea.textSelection( 'getCaretPosition', { startAndEnd: true } );
|
2010-09-15 03:08:35 +00:00
|
|
|
// Encapsulate the textarea with some containers for layout
|
|
|
|
context.$textarea
|
|
|
|
/* Disabling our loading div for now
|
|
|
|
.after( $loader )
|
|
|
|
.add( $loader )
|
|
|
|
*/
|
2012-07-03 05:52:27 +00:00
|
|
|
.wrapAll( $( '<div>' ).addClass( 'wikiEditor-ui' ) )
|
|
|
|
.wrapAll( $( '<div>' ).addClass( 'wikiEditor-ui-view wikiEditor-ui-view-wikitext' ) )
|
|
|
|
.wrapAll( $( '<div>' ).addClass( 'wikiEditor-ui-left' ) )
|
|
|
|
.wrapAll( $( '<div>' ).addClass( 'wikiEditor-ui-bottom' ) )
|
|
|
|
.wrapAll( $( '<div>' ).addClass( 'wikiEditor-ui-text' ) );
|
2014-04-29 22:43:32 +00:00
|
|
|
// Restore scroll position after this wrapAll (tracked by mediawiki.action.edit)
|
|
|
|
context.$textarea.prop( 'scrollTop', $( '#wpScrolltop' ).val() );
|
|
|
|
// Restore focus and cursor if needed
|
|
|
|
if ( hasFocus ) {
|
|
|
|
context.$textarea.focus();
|
|
|
|
context.$textarea.textSelection( 'setSelection', { start: cursorPos[0], end: cursorPos[1] } );
|
|
|
|
}
|
|
|
|
|
2010-09-15 03:08:35 +00:00
|
|
|
// Get references to some of the newly created containers
|
|
|
|
context.$ui = context.$textarea.parent().parent().parent().parent().parent();
|
|
|
|
context.$wikitext = context.$textarea.parent().parent().parent().parent();
|
|
|
|
// Add in tab and button containers
|
|
|
|
context.$wikitext
|
|
|
|
.before(
|
2012-07-03 05:52:27 +00:00
|
|
|
$( '<div>' ).addClass( 'wikiEditor-ui-controls' )
|
|
|
|
.append( $( '<div>' ).addClass( 'wikiEditor-ui-tabs' ).hide() )
|
|
|
|
.append( $( '<div>' ).addClass( 'wikiEditor-ui-buttons' ) )
|
2010-09-15 03:08:35 +00:00
|
|
|
)
|
2014-04-08 14:38:37 +00:00
|
|
|
.before( $( '<div>' ).addClass( 'wikiEditor-ui-clear' ) );
|
2010-09-15 03:08:35 +00:00
|
|
|
// Get references to some of the newly created containers
|
|
|
|
context.$controls = context.$ui.find( '.wikiEditor-ui-buttons' ).hide();
|
|
|
|
context.$buttons = context.$ui.find( '.wikiEditor-ui-buttons' );
|
|
|
|
context.$tabs = context.$ui.find( '.wikiEditor-ui-tabs' );
|
|
|
|
// Clear all floating after the UI
|
2014-04-08 14:38:37 +00:00
|
|
|
context.$ui.after( $( '<div>' ).addClass( 'wikiEditor-ui-clear' ) );
|
2010-09-15 03:08:35 +00:00
|
|
|
// Attach a right container
|
2012-07-03 05:52:27 +00:00
|
|
|
context.$wikitext.append( $( '<div>' ).addClass( 'wikiEditor-ui-right' ) );
|
2014-04-08 14:38:37 +00:00
|
|
|
context.$wikitext.append( $( '<div>' ).addClass( 'wikiEditor-ui-clear' ) );
|
2010-09-15 03:08:35 +00:00
|
|
|
// Attach a top container to the left pane
|
2012-07-03 05:52:27 +00:00
|
|
|
context.$wikitext.find( '.wikiEditor-ui-left' ).prepend( $( '<div>' ).addClass( 'wikiEditor-ui-top' ) );
|
2010-09-15 03:08:35 +00:00
|
|
|
// Setup the intial view
|
|
|
|
context.view = 'wikitext';
|
|
|
|
// Trigger the "resize" event anytime the window is resized
|
2012-07-03 05:52:27 +00:00
|
|
|
$( window ).resize( function ( event ) {
|
|
|
|
context.fn.trigger( 'resize', event );
|
|
|
|
} );
|
2010-09-15 03:08:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* API Execution */
|
|
|
|
|
|
|
|
// Since javascript gives arguments as an object, we need to convert them so they can be used more easily
|
|
|
|
var args = $.makeArray( arguments );
|
|
|
|
|
2010-10-04 20:12:30 +00:00
|
|
|
// Dynamically setup core extensions for modules that are required
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( args[0] === 'addModule' && typeof args[1] !== 'undefined' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
var modules = args[1];
|
2013-11-07 00:38:43 +00:00
|
|
|
if ( typeof modules !== 'object' ) {
|
2010-09-15 03:08:35 +00:00
|
|
|
modules = {};
|
|
|
|
modules[args[1]] = '';
|
|
|
|
}
|
2011-02-08 07:04:12 +00:00
|
|
|
for ( var module in modules ) {
|
2010-10-04 20:12:30 +00:00
|
|
|
// Only allow modules which are supported (and thus actually being turned on) affect the decision to extend
|
|
|
|
if ( module in $.wikiEditor.modules && $.wikiEditor.isSupported( $.wikiEditor.modules[module] ) ) {
|
|
|
|
// Activate all required core extensions on context
|
2011-02-08 07:04:12 +00:00
|
|
|
for ( var e in $.wikiEditor.extensions ) {
|
2010-10-04 20:12:30 +00:00
|
|
|
if (
|
|
|
|
$.wikiEditor.isRequired( $.wikiEditor.modules[module], e ) &&
|
2011-06-09 00:20:13 +00:00
|
|
|
$.inArray( e, context.extensions ) === -1
|
2010-10-04 20:12:30 +00:00
|
|
|
) {
|
|
|
|
context.extensions[context.extensions.length] = e;
|
2012-07-03 05:52:27 +00:00
|
|
|
$.wikiEditor.extensions[e]( context );
|
2010-10-04 20:12:30 +00:00
|
|
|
}
|
|
|
|
}
|
2010-09-15 03:08:35 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// There would need to be some arguments if the API is being called
|
|
|
|
if ( args.length > 0 ) {
|
|
|
|
// Handle API calls
|
|
|
|
var call = args.shift();
|
|
|
|
if ( call in context.api ) {
|
2012-07-03 05:52:27 +00:00
|
|
|
context.api[call]( context, typeof args[0] === 'undefined' ? {} : args[0] );
|
2010-09-15 03:08:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store the context for next time, and support chaining
|
2014-04-30 10:59:01 +00:00
|
|
|
return $( this ).data( 'wikiEditor-context', context );
|
2010-09-15 03:08:35 +00:00
|
|
|
|
2012-07-03 05:52:27 +00:00
|
|
|
};
|
|
|
|
|
2013-11-07 00:38:43 +00:00
|
|
|
}( jQuery, mediaWiki ) );
|