(function() { var Event = tinymce.dom.Event, each = tinymce.each, DOM = tinymce.DOM; /** * This plugin adds a left-click context menu to links in the TinyMCE editor for Zotero. * Code adopted and modified from TinyMCE contextmenu plugin. * * @class tinymce.plugins.LinksMenu */ tinymce.create('tinymce.plugins.LinksMenu', { /** * Initializes the plugin, this will be executed after the plugin has been created. * This call is done before the editor instance has finished it's initialization so use the onInit event * of the editor instance to intercept that event. * * @method init * @param {tinymce.Editor} ed Editor instance that the plugin is initialized in. * @param {string} url Absolute URL to where the plugin is located. */ init : function(ed) { var t = this, showMenu, contextmenuNeverUseNative, realCtrlKey, hideMenu; t.editor = ed; contextmenuNeverUseNative = ed.settings.contextmenu_never_use_native; // add editor command to open links through zoteroHandleEvent ed.addCommand('openlink', function(command) { var ed = tinyMCE.activeEditor; var node = ed.selection.getNode(); if (node.nodeName == 'A') { zoteroHandleEvent({ type: 'openlink', target: node, // We don't seem to be able to access the click event that triggered this // command in order to check the modifier keys used, so instead we save // the keys on every menu click in tiny_mce.js and pass them on here // for use by loadURI(). modifierKeys: ed.lastClickModifierKeys }); } }); /** * This event gets fired when the context menu is shown. * * @event onClick * @param {tinymce.plugins.LinksMenu} sender Plugin instance sending the event. * @param {tinymce.ui.DropMenu} menu Drop down menu to fill with more items if needed. */ t.onClick = new tinymce.util.Dispatcher(this); hideMenu = function(e) { hide(ed, e); }; showMenu = ed.onClick.add(function(ed, e) { // Only show on left-click if (e.button != 0) { return; } // Only show when node if (e.target.nodeName != 'A') { return; } // Block TinyMCE menu on ctrlKey and work around Safari issue if ((realCtrlKey !== 0 ? realCtrlKey : e.ctrlKey) && !contextmenuNeverUseNative) { return; } Event.cancel(e); t._getMenu(ed).showMenu(e.clientX || e.pageX, e.clientY || e.pageY); Event.add(ed.getDoc(), 'click', hideMenu); ed.nodeChanged(); }); ed.onRemove.add(function() { if (t._menu) t._menu.removeAll(); }); function hide(ed, e) { realCtrlKey = 0; // Since the contextmenu event moves // the selection we need to store it away if (e && e.button == 2) { realCtrlKey = e.ctrlKey; return; } if (t._menu) { t._menu.removeAll(); t._menu.destroy(); Event.remove(ed.getDoc(), 'click', hideMenu); t._menu = null; } }; ed.onMouseDown.add(hide); ed.onKeyDown.add(hide); ed.onKeyDown.add(function(ed, e) { if (e.shiftKey && !e.ctrlKey && !e.altKey && e.keyCode === 121) { Event.cancel(e); showMenu(ed, e); } }); }, /** * Returns information about the plugin as a name/value array. * The current keys are longname, author, authorurl, infourl and version. * * @method getInfo * @return {Object} Name/value array containing information about the plugin. */ getInfo : function() { return { longname : 'Linksmenu', author : '', authorurl : '', infourl : '', version : tinymce.majorVersion + "." + tinymce.minorVersion }; }, _getMenu : function(ed) { var t = this, m = t._menu, se = ed.selection, col = se.isCollapsed(), el = se.getNode() || ed.getBody(), am, p; if (m) { m.removeAll(); m.destroy(); } p = DOM.getPos(ed.getContentAreaContainer()); m = ed.controlManager.createDropMenu('linksmenu', { offset_x : p.x + ed.getParam('contextmenu_offset_x', 0), offset_y : p.y + ed.getParam('contextmenu_offset_y', 0), constrain : 1, keyboard_focus: true }); t._menu = m; m.add({ title : 'Open Link', icon : 'link', cmd : 'openlink', ui : true }); m.add({ title : 'Edit Link', icon : 'link', cmd : ed.plugins.advlink ? 'mceAdvLink' : 'mceLink', ui : true }); m.add({ title : 'advanced.unlink_desc', icon : 'unlink', cmd : 'UnLink' }); t.onClick.dispatch(t, m, el, col); return m; } }); // Register plugin tinymce.PluginManager.add('linksmenu', tinymce.plugins.LinksMenu); })();