/**
* Right Context Menu configuration
*
* Some usefull variables:
* node.hns = headpage id;
* node.isdir = node is namespace;
* node.dokuid = the DW id (namespace parent in case of headpage);
* id = the DW id of the selected node (headpage id in case of headpage);
* index.config.urlbase = Url Base;
* index.config.sepchar = Url separator;
*
* HOWTO EDIT:
*
* To override menu entries or add a menu entry:
* - PLEASE EDIT ONLY the scripts/contextmenu.local.js file
* - DON'T EDIT this file, it is overwritten at plugin update
*
* Base structure of the context menu is displayed below.
* The entries with 'pg' are shown for page noded, these with 'ns' only for namespaces.
*
* Current available for everybody:
* indexmenu_contextmenu['all']['pg']['view'] = [...array with menu description here... ];
* indexmenu_contextmenu['all']['pg']['edit'] = [ ... ];
* indexmenu_contextmenu['all']['ns']['view'] = [ ... ];
*
* Current available for admins:
* indexmenu_contextmenu['pg']['view'] = [ ... ];
* indexmenu_contextmenu['ns']['view'] = [ ... ];
*
* Current available for authenticated users:
* indexmenu_contextmenu['pg']['view'] = [ ... ];
* indexmenu_contextmenu['ns']['view'] = [ ... ];
*
* A menu description may contain four kind of entries:
* - section title: array with one entry e.g.:
* ['Section title (html allowed)']
* - menu action: array with two entries e.g.:
* ['Title of action 1 (html allowed)', 'javascript here ... see for examples scripts/contextmenu.js']
* - menu action with custom tooltip: array with three entries e.g.:
* ['Title of action 1 (html allowed)', 'javascript here ... see for examples scripts/contextmenu.js', 'Customized title']
* - submenu: array with two entries where second entry is an array that describes again a menu e.g.:
* ['title of submenu (html allowed)', [ ...array with menu actions... ]]
*
*
* Examples:
* A menu description array:
* ... = [
* ['section title'],
* ['title of action 1', 'javascript here'],
* ['title of submenu', [['title of subaction 1', 'javascript here'], ['title of subaction 1', 'javascript here', 'Click here for action']] ]
* ];
*
* To Override the common menu title:
* indexmenu_contextmenu['all']['pg']['view'][0] = ['customtitle'];
*
* To override a menu entry, for example the menu title:
* indexmenu_contextmenu['all']['pg']['view'][0] = ['Custom Title'];
*
* To add option to page menu:
* Array.splice(index, howManyToRemove, description1)
* index = position to start (start counting at zero)
* howManyToRemove = number of elements that are removed (set to 1 to replace a element)
* description1 = array with menu entry description
* -> optional: description2 = optional you can add more elements at once by splice(index, howManyToRemove, description1, description2, etc)
*
* indexmenu_contextmenu['all']['pg']['view'].splice(1, 0, ['Input new page', '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\');"']);
*/
/* global LANG */
/* global DOKU_BASE */
/* global JSINFO */
// IMPORTANT: DON'T MODIFY THIS FILE, BUT EDIT contextmenu.local.js PLEASE!
// THIS FILE IS OVERWRITTEN WHEN PLUGIN IS UPDATED
/**
* Right Context Menu configuration for all users:
*/
indexmenu_contextmenu['all']['pg'] = {
'view': [
[''],
[LANG.plugins.indexmenu.revs, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=revisions"'],
[LANG.plugins.indexmenu.tocpreview, '"javascript: IndexmenuContextmenu.createTocMenu(\'call=indexmenu&req=toc&id="+id+"\',\'picker_"+index.treeName+"\',\'s"+index.treeName+node.id+"\');"']
],
//Menu items in edit mode, when previewing
'edit': [
[''],
[LANG.plugins.indexmenu.insertdwlink, '"javascript: IndexmenuContextmenu.insertTags(\'"+id+"\',\'"+index.config.sepchar+"\');"+index.treeName+".divdisplay(\'r\',0);"', LANG.plugins.indexmenu.insertdwlinktooltip]
]
};
indexmenu_contextmenu['all']['ns'] = {
'view': [
[''],
[LANG.plugins.indexmenu.search, '"javascript: IndexmenuContextmenu.srchpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.isdir+"\',\'"+node.dokuid+"\');"', LANG.plugins.indexmenu.searchtooltip]
]
};
if (JSINFO && JSINFO.isadmin) {
/**
* Right Context Menu configuration for admin users:
*/
indexmenu_contextmenu['pg'] = {
'view': [
[LANG.plugins.indexmenu.edit, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=edit"'],
[''+LANG.plugins.indexmenu.create+'-->', [
[LANG.plugins.indexmenu.headpage, '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\',\'"+node.name+"\');"', LANG.plugins.indexmenu.headpagetooltip],
[LANG.plugins.indexmenu.startpage, 'IndexmenuContextmenu.getid(index.config.urlbase,id+index.config.sepchar+"start")+"do=edit"', LANG.plugins.indexmenu.startpagetooltip],
[LANG.plugins.indexmenu.custompage, '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\');"', LANG.plugins.indexmenu.custompagetooltip]
]],
[''+LANG.plugins.indexmenu.more+'-->', [
[LANG.plugins.indexmenu.acls, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=admin&page=acl"'],
[LANG.plugins.indexmenu.purgecache, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"purge=true"'],
[LANG.plugins.indexmenu.exporthtml, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=export_xhtml"'],
[LANG.plugins.indexmenu.exporttext, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=export_raw"']
]]
]
};
indexmenu_contextmenu['ns'] = {
'view': [
[LANG.plugins.indexmenu.newpage, '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\');"', LANG.plugins.indexmenu.newpagetooltip],
[''+LANG.plugins.indexmenu.more+'-->', [
[LANG.plugins.indexmenu.headpagehere, '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\',\'"+node.name+"\');"', LANG.plugins.indexmenu.headpageheretooltip],
[LANG.plugins.indexmenu.acls, 'IndexmenuContextmenu.getid(index.config.urlbase,node.dokuid)+"do=admin&page=acl"']
]]
]
};
} else if (JSINFO && JSINFO.isauth) {
/**
* Right Context Menu configuration for authenticated users:
*/
indexmenu_contextmenu['pg'] = {
'view': [
[LANG.plugins.indexmenu.newpagehere, '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\');"'],
['Edit', 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=edit"', 1, 0 ],
[''+LANG.plugins.indexmenu.more+'-->', [
[LANG.plugins.indexmenu.headpagehere, '"javascript: IndexmenuContextmenu.reqpage(\'"+index.config.urlbase+"\',\'"+index.config.sepchar+"\',\'"+node.dokuid+"\',\'"+node.name+"\');"'],
[LANG.plugins.indexmenu.purgecache, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"purge=true"'],
[LANG.plugins.indexmenu.exporthtml, 'IndexmenuContextmenu.getid(index.config.urlbase,id)+"do=export_xhtml"']
]]
]
};
}
var IndexmenuContextmenu = {
/**
* Common functions
* Insert your custom functions (available for all users) here.
*/
/**
* Navigate to the search page
*
* @param {string} urlbase index.config.urlbase
* @param {string} sepchar index.config.sepchar
* @param {boolean} isdir whether a directory (probably boolean, might be string)
* @param {string} nid page id of node (dokuid mentioned in indexmenu.js)
*/
srchpage: function (urlbase, sepchar, isdir, nid) {
const enteredText = prompt(LANG.plugins.indexmenu.insertkeywords, "");
if (enteredText) {
let fnid = nid;
if (isdir == "0") {
fnid = fnid.substring(0, nid.lastIndexOf(sepchar));
}
let b = urlbase, re = new RegExp(sepchar, 'g');
fnid = fnid.replace(re, ":");
b += (urlbase.indexOf("?id=") < 0) ? '?id=' : '';
window.location.href = IndexmenuContextmenu.getid(b, enteredText + " @" + fnid) + "do=search";
}
},
/**
* Build a url to a wiki page
*
* @param {string} urlbase
* @param {string} id
* @returns {string}
*/
getid: function (urlbase, id) {
let url = (urlbase || '') + encodeURIComponent(id || '');
url += (urlbase.indexOf("?") < 0) ? '?' : '&';
return url;
},
/**
* Navigate to the editor window
*
* @param {string} urlbase
* @param {string} sepchar
* @param {string} id
* @param {string} pagename
*/
reqpage: function (urlbase, sepchar, id, pagename) {
let newpageid;
if (pagename) {
newpageid = id + sepchar + pagename;
} else {
newpageid = prompt(LANG.plugins.indexmenu.insertpagename, "");
if (!newpageid) {
return;
}
newpageid = id + sepchar + newpageid;
}
if (newpageid) {
window.location.href = IndexmenuContextmenu.getid(urlbase, newpageid) + "do=edit";
}
},
/**
* Insert link syntax with given id in current editor window
*
* @param {string} lnk page id
* @param {string} sepchar
*/
insertTags: function (lnk, sepchar) {
let r, l = lnk;
if (sepchar) {
r = new RegExp(sepchar, "g");
l = lnk.replace(r, ':');
}
insertTags('wiki__text', '[[', ']]', l);
},
/**
* Create or catch the picker and hide it, next call the ajax content loading to get the ToC
*
* @param {string} get query string
* @param {string} picker id of picker
* @param {string} btn id of button
*/
createTocMenu: function (get, picker, btn) {
var $toc_picker = jQuery('#' + picker);
if (!$toc_picker.length) {
$toc_picker = IndexmenuUtils.createPicker(picker, 'indexmenu_toc');
$toc_picker
.html('
')
.children().first().click(function (event) {
event.stopPropagation();
return IndexmenuContextmenu.togglePicker($toc_picker, jQuery('#' + btn));
});
} else {
$toc_picker.hide();
}
IndexmenuContextmenu.ajaxmenu(get, $toc_picker, jQuery('#' + btn), $toc_picker.children().last(), null);
},
/**
* Shows the picker and adds to it or to an internal containter the ajax content
*
* @param {string} get query string
* @param {jQuery} $picker
* @param {jQuery} $btn
* @param {jQuery} $container if defined ajax result is added to it, otherwise to $picker
* @param {function} oncomplete called when defined to handle ajax result
*/
ajaxmenu: function (get, $picker, $btn, $container, oncomplete) {
var $indx_list;
$indx_list = $container || $picker;
if (!IndexmenuContextmenu.togglePicker($picker, $btn)) return;
var onComplete = function (data) {
$indx_list.html('');
if (typeof oncomplete == 'function') {
oncomplete(data, $indx_list);
} else {
$indx_list.html(data);
}
};
//get content for picker/container
jQuery.ajax({
type: "POST",
url: DOKU_BASE + 'lib/exe/ajax.php',
data: get,
beforeSend: function () {
$indx_list.html('');
},
success: onComplete,
dataType: 'html'
});
},
/**
* Hide/show picker, will be shown beside btn
*
* @param {string|jQuery} $picker
* @param {jQuery} $btn
* @return {Boolean} true if open, false closed
*/
togglePicker: function ($picker, $btn) {
var x = 8, y = 0;
if (!$picker.is(':visible')) {
var pos = $btn.offset();
//position + width of button
x += pos.left + $btn[0].offsetWidth;
y += pos.top;
$picker
.show()
.offset({
left: x,
top: y
});
return true;
} else {
$picker.hide();
return false;
}
},
/**
* Fills the contextmenu by creating entries from the given configuration arrays and concatenating these
* to the #r picker
*
* @param {any[]} amenu (part of) the configuration array
* @param {dTree} index the indexmenu object
* @param {int} n node id
*/
arrconcat: function (amenu, index, n) {
var html, id, item, a, li;
if (typeof amenu == 'undefined' || typeof amenu['view'] == 'undefined') {
return;
}
var cmenu = amenu['view'];
if (jQuery('#tool__bar')[0] && amenu['edit'] instanceof Array) {
cmenu = amenu['edit'].concat(cmenu);
}
var node = index.aNodes[n];
id = node.hns || node.dokuid;
var createCMenuEntry = function (entry) {
return '' + entry[0] + '';
};
jQuery.each(cmenu, function (i, cmenuentry) {
if (cmenuentry == '') {
return true;
}
item = document.createElement('li');
var $cmenu = jQuery('#r' + index.treeName);
if (cmenuentry[1]) {
if (cmenuentry[1] instanceof Array) {
html = document.createElement('ul');
jQuery.each(cmenuentry[1], function (a, subcmenuentry) {
li = document.createElement('li');
li.innerHTML = createCMenuEntry(subcmenuentry);
html.appendChild(li);
});
//}
item.innerHTML = '';
html.left = $cmenu[0].width;
item.appendChild(html);
} else {
item.innerHTML = createCMenuEntry(cmenuentry);
}
} else {
item.innerHTML = cmenuentry;
}
$cmenu.children().last().append(item);
});
},
/**
* Absolute positioning of the div at place of mouseclick
*
* @param obj div element
* @param e
*/
mouseposition: function (obj, e) {
//http://www.quirksmode.org/js/events_properties.html
var X = 0, Y = 0;
if (!e) e = window.event;
if (e.pageX || e.pageY) {
X = e.pageX;
Y = e.pageY;
}
else if (e.clientX || e.clientY) {
X = e.clientX + document.body.scrollLeft + document.documentElement.scrollLeft;
Y = e.clientY + document.body.scrollTop + document.documentElement.scrollTop;
}
obj.style.left = X - 5 + 'px';
obj.style.top = Y - 5 + 'px';
},
/**
* Check mouse button onmousedown event, only for middle and right mouse button contextmenu is shown
*
* @param {int} n node id
* @param {string|dTree} obj the unique name of a dTree object
* @param {event} e
*/
checkcontextm: function (n, obj, e) {
e = e || event;
// mouse clicks: which 3 === right, button 2 === right button
if ((e.which === 3 || e.button === 2) || (window.opera && e.which === 1 && e.ctrlKey)) {
obj.contextmenu(n, e);
IndexmenuContextmenu.stopevt(e);
}
},
/**
* Prevent default oncontextmenu event
*
* @param {event} e
* @returns {boolean}
*/
stopevt: function (e) {
if (!window.indexmenu_contextmenu) {
return true;
}
e = e || event;
e.preventDefault ? e.preventDefault() : e.returnValue = false;
return false;
}
};