1195 lines
31 KiB
JavaScript
1195 lines
31 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
|
||
|
var Highlighter = require('./Highlighter');
|
||
|
var History = require('./History');
|
||
|
var SearchBox = require('./SearchBox');
|
||
|
var ContextMenu = require('./ContextMenu');
|
||
|
var Node = require('./Node');
|
||
|
var ModeSwitcher = require('./ModeSwitcher');
|
||
|
var util = require('./util');
|
||
|
|
||
|
// create a mixin with the functions for tree mode
|
||
|
var treemode = {};
|
||
|
|
||
|
/**
|
||
|
* Create a tree editor
|
||
|
* @param {Element} container Container element
|
||
|
* @param {Object} [options] Object with options. available options:
|
||
|
* {String} mode Editor mode. Available values:
|
||
|
* 'tree' (default), 'view',
|
||
|
* and 'form'.
|
||
|
* {Boolean} search Enable search box.
|
||
|
* True by default
|
||
|
* {Boolean} history Enable history (undo/redo).
|
||
|
* True by default
|
||
|
* {function} onChange Callback method, triggered
|
||
|
* on change of contents
|
||
|
* {String} name Field name for the root node.
|
||
|
* {boolean} escapeUnicode If true, unicode
|
||
|
* characters are escaped.
|
||
|
* false by default.
|
||
|
* {Object} schema A JSON Schema for validation
|
||
|
* @private
|
||
|
*/
|
||
|
treemode.create = function (container, options) {
|
||
|
if (!container) {
|
||
|
throw new Error('No container element provided.');
|
||
|
}
|
||
|
this.container = container;
|
||
|
this.dom = {};
|
||
|
this.highlighter = new Highlighter();
|
||
|
this.selection = undefined; // will hold the last input selection
|
||
|
this.multiselection = {
|
||
|
nodes: []
|
||
|
};
|
||
|
this.validateSchema = null; // will be set in .setSchema(schema)
|
||
|
this.errorNodes = [];
|
||
|
|
||
|
this.node = null;
|
||
|
this.focusTarget = null;
|
||
|
|
||
|
this._setOptions(options);
|
||
|
|
||
|
if (this.options.history && this.options.mode !== 'view') {
|
||
|
this.history = new History(this);
|
||
|
}
|
||
|
|
||
|
this._createFrame();
|
||
|
this._createTable();
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Destroy the editor. Clean up DOM, event listeners, and web workers.
|
||
|
*/
|
||
|
treemode.destroy = function () {
|
||
|
if (this.frame && this.container && this.frame.parentNode == this.container) {
|
||
|
this.container.removeChild(this.frame);
|
||
|
this.frame = null;
|
||
|
}
|
||
|
this.container = null;
|
||
|
|
||
|
this.dom = null;
|
||
|
|
||
|
this.clear();
|
||
|
this.node = null;
|
||
|
this.focusTarget = null;
|
||
|
this.selection = null;
|
||
|
this.multiselection = null;
|
||
|
this.errorNodes = null;
|
||
|
this.validateSchema = null;
|
||
|
this._debouncedValidate = null;
|
||
|
|
||
|
if (this.history) {
|
||
|
this.history.destroy();
|
||
|
this.history = null;
|
||
|
}
|
||
|
|
||
|
if (this.searchBox) {
|
||
|
this.searchBox.destroy();
|
||
|
this.searchBox = null;
|
||
|
}
|
||
|
|
||
|
if (this.modeSwitcher) {
|
||
|
this.modeSwitcher.destroy();
|
||
|
this.modeSwitcher = null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Initialize and set default options
|
||
|
* @param {Object} [options] See description in constructor
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._setOptions = function (options) {
|
||
|
this.options = {
|
||
|
search: true,
|
||
|
history: true,
|
||
|
mode: 'tree',
|
||
|
name: undefined, // field name of root node
|
||
|
schema: null
|
||
|
};
|
||
|
|
||
|
// copy all options
|
||
|
if (options) {
|
||
|
for (var prop in options) {
|
||
|
if (options.hasOwnProperty(prop)) {
|
||
|
this.options[prop] = options[prop];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// compile a JSON schema validator if a JSON schema is provided
|
||
|
this.setSchema(this.options.schema);
|
||
|
|
||
|
// create a debounced validate function
|
||
|
this._debouncedValidate = util.debounce(this.validate.bind(this), this.DEBOUNCE_INTERVAL);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set JSON object in editor
|
||
|
* @param {Object | undefined} json JSON data
|
||
|
* @param {String} [name] Optional field name for the root node.
|
||
|
* Can also be set using setName(name).
|
||
|
*/
|
||
|
treemode.set = function (json, name) {
|
||
|
// adjust field name for root node
|
||
|
if (name) {
|
||
|
// TODO: deprecated since version 2.2.0. Cleanup some day.
|
||
|
console.warn('Second parameter "name" is deprecated. Use setName(name) instead.');
|
||
|
this.options.name = name;
|
||
|
}
|
||
|
|
||
|
// verify if json is valid JSON, ignore when a function
|
||
|
if (json instanceof Function || (json === undefined)) {
|
||
|
this.clear();
|
||
|
}
|
||
|
else {
|
||
|
this.content.removeChild(this.table); // Take the table offline
|
||
|
|
||
|
// replace the root node
|
||
|
var params = {
|
||
|
field: this.options.name,
|
||
|
value: json
|
||
|
};
|
||
|
var node = new Node(this, params);
|
||
|
this._setRoot(node);
|
||
|
|
||
|
// validate JSON schema (if configured)
|
||
|
this.validate();
|
||
|
|
||
|
// expand
|
||
|
var recurse = false;
|
||
|
this.node.expand(recurse);
|
||
|
|
||
|
this.content.appendChild(this.table); // Put the table online again
|
||
|
}
|
||
|
|
||
|
// TODO: maintain history, store last state and previous document
|
||
|
if (this.history) {
|
||
|
this.history.clear();
|
||
|
}
|
||
|
|
||
|
// clear search
|
||
|
if (this.searchBox) {
|
||
|
this.searchBox.clear();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Get JSON object from editor
|
||
|
* @return {Object | undefined} json
|
||
|
*/
|
||
|
treemode.get = function () {
|
||
|
// remove focus from currently edited node
|
||
|
if (this.focusTarget) {
|
||
|
var node = Node.getNodeFromTarget(this.focusTarget);
|
||
|
if (node) {
|
||
|
node.blur();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (this.node) {
|
||
|
return this.node.getValue();
|
||
|
}
|
||
|
else {
|
||
|
return undefined;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Get the text contents of the editor
|
||
|
* @return {String} jsonText
|
||
|
*/
|
||
|
treemode.getText = function() {
|
||
|
return JSON.stringify(this.get());
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set the text contents of the editor
|
||
|
* @param {String} jsonText
|
||
|
*/
|
||
|
treemode.setText = function(jsonText) {
|
||
|
this.set(util.parse(jsonText));
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set a field name for the root node.
|
||
|
* @param {String | undefined} name
|
||
|
*/
|
||
|
treemode.setName = function (name) {
|
||
|
this.options.name = name;
|
||
|
if (this.node) {
|
||
|
this.node.updateField(this.options.name);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Get the field name for the root node.
|
||
|
* @return {String | undefined} name
|
||
|
*/
|
||
|
treemode.getName = function () {
|
||
|
return this.options.name;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set focus to the editor. Focus will be set to:
|
||
|
* - the first editable field or value, or else
|
||
|
* - to the expand button of the root node, or else
|
||
|
* - to the context menu button of the root node, or else
|
||
|
* - to the first button in the top menu
|
||
|
*/
|
||
|
treemode.focus = function () {
|
||
|
var input = this.content.querySelector('[contenteditable=true]');
|
||
|
if (input) {
|
||
|
input.focus();
|
||
|
}
|
||
|
else if (this.node.dom.expand) {
|
||
|
this.node.dom.expand.focus();
|
||
|
}
|
||
|
else if (this.node.dom.menu) {
|
||
|
this.node.dom.menu.focus();
|
||
|
}
|
||
|
else {
|
||
|
// focus to the first button in the menu
|
||
|
input = this.frame.querySelector('button');
|
||
|
if (input) {
|
||
|
input.focus();
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Remove the root node from the editor
|
||
|
*/
|
||
|
treemode.clear = function () {
|
||
|
if (this.node) {
|
||
|
this.node.collapse();
|
||
|
this.tbody.removeChild(this.node.getDom());
|
||
|
delete this.node;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set the root node for the json editor
|
||
|
* @param {Node} node
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._setRoot = function (node) {
|
||
|
this.clear();
|
||
|
|
||
|
this.node = node;
|
||
|
|
||
|
// append to the dom
|
||
|
this.tbody.appendChild(node.getDom());
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Search text in all nodes
|
||
|
* The nodes will be expanded when the text is found one of its childs,
|
||
|
* else it will be collapsed. Searches are case insensitive.
|
||
|
* @param {String} text
|
||
|
* @return {Object[]} results Array with nodes containing the search results
|
||
|
* The result objects contains fields:
|
||
|
* - {Node} node,
|
||
|
* - {String} elem the dom element name where
|
||
|
* the result is found ('field' or
|
||
|
* 'value')
|
||
|
*/
|
||
|
treemode.search = function (text) {
|
||
|
var results;
|
||
|
if (this.node) {
|
||
|
this.content.removeChild(this.table); // Take the table offline
|
||
|
results = this.node.search(text);
|
||
|
this.content.appendChild(this.table); // Put the table online again
|
||
|
}
|
||
|
else {
|
||
|
results = [];
|
||
|
}
|
||
|
|
||
|
return results;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Expand all nodes
|
||
|
*/
|
||
|
treemode.expandAll = function () {
|
||
|
if (this.node) {
|
||
|
this.content.removeChild(this.table); // Take the table offline
|
||
|
this.node.expand();
|
||
|
this.content.appendChild(this.table); // Put the table online again
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Collapse all nodes
|
||
|
*/
|
||
|
treemode.collapseAll = function () {
|
||
|
if (this.node) {
|
||
|
this.content.removeChild(this.table); // Take the table offline
|
||
|
this.node.collapse();
|
||
|
this.content.appendChild(this.table); // Put the table online again
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* The method onChange is called whenever a field or value is changed, created,
|
||
|
* deleted, duplicated, etc.
|
||
|
* @param {String} action Change action. Available values: "editField",
|
||
|
* "editValue", "changeType", "appendNode",
|
||
|
* "removeNode", "duplicateNode", "moveNode", "expand",
|
||
|
* "collapse".
|
||
|
* @param {Object} params Object containing parameters describing the change.
|
||
|
* The parameters in params depend on the action (for
|
||
|
* example for "editValue" the Node, old value, and new
|
||
|
* value are provided). params contains all information
|
||
|
* needed to undo or redo the action.
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onAction = function (action, params) {
|
||
|
// add an action to the history
|
||
|
if (this.history) {
|
||
|
this.history.add(action, params);
|
||
|
}
|
||
|
|
||
|
this._onChange();
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Handle a change:
|
||
|
* - Validate JSON schema
|
||
|
* - Send a callback to the onChange listener if provided
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onChange = function () {
|
||
|
// validate JSON schema (if configured)
|
||
|
this._debouncedValidate();
|
||
|
|
||
|
// trigger the onChange callback
|
||
|
if (this.options.onChange) {
|
||
|
try {
|
||
|
this.options.onChange();
|
||
|
}
|
||
|
catch (err) {
|
||
|
console.error('Error in onChange callback: ', err);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Validate current JSON object against the configured JSON schema
|
||
|
* Throws an exception when no JSON schema is configured
|
||
|
*/
|
||
|
treemode.validate = function () {
|
||
|
// clear all current errors
|
||
|
if (this.errorNodes) {
|
||
|
this.errorNodes.forEach(function (node) {
|
||
|
node.setError(null);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
var root = this.node;
|
||
|
if (!root) { // TODO: this should be redundant but is needed on mode switch
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// check for duplicate keys
|
||
|
var duplicateErrors = root.validate();
|
||
|
|
||
|
// validate the JSON
|
||
|
var schemaErrors = [];
|
||
|
if (this.validateSchema) {
|
||
|
var valid = this.validateSchema(root.getValue());
|
||
|
if (!valid) {
|
||
|
// apply all new errors
|
||
|
schemaErrors = this.validateSchema.errors
|
||
|
.map(function (error) {
|
||
|
return util.improveSchemaError(error);
|
||
|
})
|
||
|
.map(function findNode (error) {
|
||
|
return {
|
||
|
node: root.findNode(error.dataPath),
|
||
|
error: error
|
||
|
}
|
||
|
})
|
||
|
.filter(function hasNode (entry) {
|
||
|
return entry.node != null
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// display the error in the nodes with a problem
|
||
|
this.errorNodes = duplicateErrors
|
||
|
.concat(schemaErrors)
|
||
|
.reduce(function expandParents (all, entry) {
|
||
|
// expand parents, then merge such that parents come first and
|
||
|
// original entries last
|
||
|
return entry.node
|
||
|
.findParents()
|
||
|
.map(function (parent) {
|
||
|
return {
|
||
|
node: parent,
|
||
|
child: entry.node,
|
||
|
error: {
|
||
|
message: parent.type === 'object'
|
||
|
? 'Contains invalid properties' // object
|
||
|
: 'Contains invalid items' // array
|
||
|
}
|
||
|
};
|
||
|
})
|
||
|
.concat(all, [entry]);
|
||
|
}, [])
|
||
|
// TODO: dedupe the parent nodes
|
||
|
.map(function setError (entry) {
|
||
|
entry.node.setError(entry.error, entry.child);
|
||
|
return entry.node;
|
||
|
});
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Refresh the rendered contents
|
||
|
*/
|
||
|
treemode.refresh = function () {
|
||
|
if (this.node) {
|
||
|
this.node.updateDom({recurse: true});
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Start autoscrolling when given mouse position is above the top of the
|
||
|
* editor contents, or below the bottom.
|
||
|
* @param {Number} mouseY Absolute mouse position in pixels
|
||
|
*/
|
||
|
treemode.startAutoScroll = function (mouseY) {
|
||
|
var me = this;
|
||
|
var content = this.content;
|
||
|
var top = util.getAbsoluteTop(content);
|
||
|
var height = content.clientHeight;
|
||
|
var bottom = top + height;
|
||
|
var margin = 24;
|
||
|
var interval = 50; // ms
|
||
|
|
||
|
if ((mouseY < top + margin) && content.scrollTop > 0) {
|
||
|
this.autoScrollStep = ((top + margin) - mouseY) / 3;
|
||
|
}
|
||
|
else if (mouseY > bottom - margin &&
|
||
|
height + content.scrollTop < content.scrollHeight) {
|
||
|
this.autoScrollStep = ((bottom - margin) - mouseY) / 3;
|
||
|
}
|
||
|
else {
|
||
|
this.autoScrollStep = undefined;
|
||
|
}
|
||
|
|
||
|
if (this.autoScrollStep) {
|
||
|
if (!this.autoScrollTimer) {
|
||
|
this.autoScrollTimer = setInterval(function () {
|
||
|
if (me.autoScrollStep) {
|
||
|
content.scrollTop -= me.autoScrollStep;
|
||
|
}
|
||
|
else {
|
||
|
me.stopAutoScroll();
|
||
|
}
|
||
|
}, interval);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
this.stopAutoScroll();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Stop auto scrolling. Only applicable when scrolling
|
||
|
*/
|
||
|
treemode.stopAutoScroll = function () {
|
||
|
if (this.autoScrollTimer) {
|
||
|
clearTimeout(this.autoScrollTimer);
|
||
|
delete this.autoScrollTimer;
|
||
|
}
|
||
|
if (this.autoScrollStep) {
|
||
|
delete this.autoScrollStep;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Set the focus to an element in the editor, set text selection, and
|
||
|
* set scroll position.
|
||
|
* @param {Object} selection An object containing fields:
|
||
|
* {Element | undefined} dom The dom element
|
||
|
* which has focus
|
||
|
* {Range | TextRange} range A text selection
|
||
|
* {Node[]} nodes Nodes in case of multi selection
|
||
|
* {Number} scrollTop Scroll position
|
||
|
*/
|
||
|
treemode.setSelection = function (selection) {
|
||
|
if (!selection) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ('scrollTop' in selection && this.content) {
|
||
|
// TODO: animated scroll
|
||
|
this.content.scrollTop = selection.scrollTop;
|
||
|
}
|
||
|
if (selection.nodes) {
|
||
|
// multi-select
|
||
|
this.select(selection.nodes);
|
||
|
}
|
||
|
if (selection.range) {
|
||
|
util.setSelectionOffset(selection.range);
|
||
|
}
|
||
|
if (selection.dom) {
|
||
|
selection.dom.focus();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Get the current focus
|
||
|
* @return {Object} selection An object containing fields:
|
||
|
* {Element | undefined} dom The dom element
|
||
|
* which has focus
|
||
|
* {Range | TextRange} range A text selection
|
||
|
* {Node[]} nodes Nodes in case of multi selection
|
||
|
* {Number} scrollTop Scroll position
|
||
|
*/
|
||
|
treemode.getSelection = function () {
|
||
|
var range = util.getSelectionOffset();
|
||
|
if (range && range.container.nodeName !== 'DIV') { // filter on (editable) divs)
|
||
|
range = null;
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
dom: this.focusTarget,
|
||
|
range: range,
|
||
|
nodes: this.multiselection.nodes.slice(0),
|
||
|
scrollTop: this.content ? this.content.scrollTop : 0
|
||
|
};
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Adjust the scroll position such that given top position is shown at 1/4
|
||
|
* of the window height.
|
||
|
* @param {Number} top
|
||
|
* @param {function(boolean)} [callback] Callback, executed when animation is
|
||
|
* finished. The callback returns true
|
||
|
* when animation is finished, or false
|
||
|
* when not.
|
||
|
*/
|
||
|
treemode.scrollTo = function (top, callback) {
|
||
|
var content = this.content;
|
||
|
if (content) {
|
||
|
var editor = this;
|
||
|
// cancel any running animation
|
||
|
if (editor.animateTimeout) {
|
||
|
clearTimeout(editor.animateTimeout);
|
||
|
delete editor.animateTimeout;
|
||
|
}
|
||
|
if (editor.animateCallback) {
|
||
|
editor.animateCallback(false);
|
||
|
delete editor.animateCallback;
|
||
|
}
|
||
|
|
||
|
// calculate final scroll position
|
||
|
var height = content.clientHeight;
|
||
|
var bottom = content.scrollHeight - height;
|
||
|
var finalScrollTop = Math.min(Math.max(top - height / 4, 0), bottom);
|
||
|
|
||
|
// animate towards the new scroll position
|
||
|
var animate = function () {
|
||
|
var scrollTop = content.scrollTop;
|
||
|
var diff = (finalScrollTop - scrollTop);
|
||
|
if (Math.abs(diff) > 3) {
|
||
|
content.scrollTop += diff / 3;
|
||
|
editor.animateCallback = callback;
|
||
|
editor.animateTimeout = setTimeout(animate, 50);
|
||
|
}
|
||
|
else {
|
||
|
// finished
|
||
|
if (callback) {
|
||
|
callback(true);
|
||
|
}
|
||
|
content.scrollTop = finalScrollTop;
|
||
|
delete editor.animateTimeout;
|
||
|
delete editor.animateCallback;
|
||
|
}
|
||
|
};
|
||
|
animate();
|
||
|
}
|
||
|
else {
|
||
|
if (callback) {
|
||
|
callback(false);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Create main frame
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._createFrame = function () {
|
||
|
// create the frame
|
||
|
this.frame = document.createElement('div');
|
||
|
this.frame.className = 'jsoneditor jsoneditor-mode-' + this.options.mode;
|
||
|
this.container.appendChild(this.frame);
|
||
|
|
||
|
// create one global event listener to handle all events from all nodes
|
||
|
var editor = this;
|
||
|
function onEvent(event) {
|
||
|
// when switching to mode "code" or "text" via the menu, some events
|
||
|
// are still fired whilst the _onEvent methods is already removed.
|
||
|
if (editor._onEvent) {
|
||
|
editor._onEvent(event);
|
||
|
}
|
||
|
}
|
||
|
this.frame.onclick = function (event) {
|
||
|
var target = event.target;// || event.srcElement;
|
||
|
|
||
|
onEvent(event);
|
||
|
|
||
|
// prevent default submit action of buttons when editor is located
|
||
|
// inside a form
|
||
|
if (target.nodeName == 'BUTTON') {
|
||
|
event.preventDefault();
|
||
|
}
|
||
|
};
|
||
|
this.frame.oninput = onEvent;
|
||
|
this.frame.onchange = onEvent;
|
||
|
this.frame.onkeydown = onEvent;
|
||
|
this.frame.onkeyup = onEvent;
|
||
|
this.frame.oncut = onEvent;
|
||
|
this.frame.onpaste = onEvent;
|
||
|
this.frame.onmousedown = onEvent;
|
||
|
this.frame.onmouseup = onEvent;
|
||
|
this.frame.onmouseover = onEvent;
|
||
|
this.frame.onmouseout = onEvent;
|
||
|
// Note: focus and blur events do not propagate, therefore they defined
|
||
|
// using an eventListener with useCapture=true
|
||
|
// see http://www.quirksmode.org/blog/archives/2008/04/delegating_the.html
|
||
|
util.addEventListener(this.frame, 'focus', onEvent, true);
|
||
|
util.addEventListener(this.frame, 'blur', onEvent, true);
|
||
|
this.frame.onfocusin = onEvent; // for IE
|
||
|
this.frame.onfocusout = onEvent; // for IE
|
||
|
|
||
|
// create menu
|
||
|
this.menu = document.createElement('div');
|
||
|
this.menu.className = 'jsoneditor-menu';
|
||
|
this.frame.appendChild(this.menu);
|
||
|
|
||
|
// create expand all button
|
||
|
var expandAll = document.createElement('button');
|
||
|
expandAll.className = 'jsoneditor-expand-all';
|
||
|
expandAll.title = 'Expand all fields';
|
||
|
expandAll.onclick = function () {
|
||
|
editor.expandAll();
|
||
|
};
|
||
|
this.menu.appendChild(expandAll);
|
||
|
|
||
|
// create expand all button
|
||
|
var collapseAll = document.createElement('button');
|
||
|
collapseAll.title = 'Collapse all fields';
|
||
|
collapseAll.className = 'jsoneditor-collapse-all';
|
||
|
collapseAll.onclick = function () {
|
||
|
editor.collapseAll();
|
||
|
};
|
||
|
this.menu.appendChild(collapseAll);
|
||
|
|
||
|
// create undo/redo buttons
|
||
|
if (this.history) {
|
||
|
// create undo button
|
||
|
var undo = document.createElement('button');
|
||
|
undo.className = 'jsoneditor-undo jsoneditor-separator';
|
||
|
undo.title = 'Undo last action (Ctrl+Z)';
|
||
|
undo.onclick = function () {
|
||
|
editor._onUndo();
|
||
|
};
|
||
|
this.menu.appendChild(undo);
|
||
|
this.dom.undo = undo;
|
||
|
|
||
|
// create redo button
|
||
|
var redo = document.createElement('button');
|
||
|
redo.className = 'jsoneditor-redo';
|
||
|
redo.title = 'Redo (Ctrl+Shift+Z)';
|
||
|
redo.onclick = function () {
|
||
|
editor._onRedo();
|
||
|
};
|
||
|
this.menu.appendChild(redo);
|
||
|
this.dom.redo = redo;
|
||
|
|
||
|
// register handler for onchange of history
|
||
|
this.history.onChange = function () {
|
||
|
undo.disabled = !editor.history.canUndo();
|
||
|
redo.disabled = !editor.history.canRedo();
|
||
|
};
|
||
|
this.history.onChange();
|
||
|
}
|
||
|
|
||
|
// create mode box
|
||
|
if (this.options && this.options.modes && this.options.modes.length) {
|
||
|
var me = this;
|
||
|
this.modeSwitcher = new ModeSwitcher(this.menu, this.options.modes, this.options.mode, function onSwitch(mode) {
|
||
|
me.modeSwitcher.destroy();
|
||
|
|
||
|
// switch mode and restore focus
|
||
|
me.setMode(mode);
|
||
|
me.modeSwitcher.focus();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// create search box
|
||
|
if (this.options.search) {
|
||
|
this.searchBox = new SearchBox(this, this.menu);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Perform an undo action
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onUndo = function () {
|
||
|
if (this.history) {
|
||
|
// undo last action
|
||
|
this.history.undo();
|
||
|
|
||
|
// fire change event
|
||
|
this._onChange();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Perform a redo action
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onRedo = function () {
|
||
|
if (this.history) {
|
||
|
// redo last action
|
||
|
this.history.redo();
|
||
|
|
||
|
// fire change event
|
||
|
this._onChange();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Event handler
|
||
|
* @param event
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onEvent = function (event) {
|
||
|
if (event.type == 'keydown') {
|
||
|
this._onKeyDown(event);
|
||
|
}
|
||
|
|
||
|
if (event.type == 'focus') {
|
||
|
this.focusTarget = event.target;
|
||
|
}
|
||
|
|
||
|
if (event.type == 'mousedown') {
|
||
|
this._startDragDistance(event);
|
||
|
}
|
||
|
if (event.type == 'mousemove' || event.type == 'mouseup' || event.type == 'click') {
|
||
|
this._updateDragDistance(event);
|
||
|
}
|
||
|
|
||
|
var node = Node.getNodeFromTarget(event.target);
|
||
|
|
||
|
if (node && node.selected) {
|
||
|
if (event.type == 'click') {
|
||
|
if (event.target == node.dom.menu) {
|
||
|
this.showContextMenu(event.target);
|
||
|
|
||
|
// stop propagation (else we will open the context menu of a single node)
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// deselect a multi selection
|
||
|
if (!event.hasMoved) {
|
||
|
this.deselect();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (event.type == 'mousedown') {
|
||
|
// drag multiple nodes
|
||
|
Node.onDragStart(this.multiselection.nodes, event);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
if (event.type == 'mousedown') {
|
||
|
this.deselect();
|
||
|
|
||
|
if (node && event.target == node.dom.drag) {
|
||
|
// drag a singe node
|
||
|
Node.onDragStart(node, event);
|
||
|
}
|
||
|
else if (!node || (event.target != node.dom.field && event.target != node.dom.value && event.target != node.dom.select)) {
|
||
|
// select multiple nodes
|
||
|
this._onMultiSelectStart(event);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (node) {
|
||
|
node.onEvent(event);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
treemode._startDragDistance = function (event) {
|
||
|
this.dragDistanceEvent = {
|
||
|
initialTarget: event.target,
|
||
|
initialPageX: event.pageX,
|
||
|
initialPageY: event.pageY,
|
||
|
dragDistance: 0,
|
||
|
hasMoved: false
|
||
|
};
|
||
|
};
|
||
|
|
||
|
treemode._updateDragDistance = function (event) {
|
||
|
if (!this.dragDistanceEvent) {
|
||
|
this._startDragDistance(event);
|
||
|
}
|
||
|
|
||
|
var diffX = event.pageX - this.dragDistanceEvent.initialPageX;
|
||
|
var diffY = event.pageY - this.dragDistanceEvent.initialPageY;
|
||
|
|
||
|
this.dragDistanceEvent.dragDistance = Math.sqrt(diffX * diffX + diffY * diffY);
|
||
|
this.dragDistanceEvent.hasMoved =
|
||
|
this.dragDistanceEvent.hasMoved || this.dragDistanceEvent.dragDistance > 10;
|
||
|
|
||
|
event.dragDistance = this.dragDistanceEvent.dragDistance;
|
||
|
event.hasMoved = this.dragDistanceEvent.hasMoved;
|
||
|
|
||
|
return event.dragDistance;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Start multi selection of nodes by dragging the mouse
|
||
|
* @param event
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onMultiSelectStart = function (event) {
|
||
|
var node = Node.getNodeFromTarget(event.target);
|
||
|
|
||
|
if (this.options.mode !== 'tree' || this.options.onEditable !== undefined) {
|
||
|
// dragging not allowed in modes 'view' and 'form'
|
||
|
// TODO: allow multiselection of items when option onEditable is specified
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
this.multiselection = {
|
||
|
start: node || null,
|
||
|
end: null,
|
||
|
nodes: []
|
||
|
};
|
||
|
|
||
|
this._startDragDistance(event);
|
||
|
|
||
|
var editor = this;
|
||
|
if (!this.mousemove) {
|
||
|
this.mousemove = util.addEventListener(window, 'mousemove', function (event) {
|
||
|
editor._onMultiSelect(event);
|
||
|
});
|
||
|
}
|
||
|
if (!this.mouseup) {
|
||
|
this.mouseup = util.addEventListener(window, 'mouseup', function (event ) {
|
||
|
editor._onMultiSelectEnd(event);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Multiselect nodes by dragging
|
||
|
* @param event
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onMultiSelect = function (event) {
|
||
|
event.preventDefault();
|
||
|
|
||
|
this._updateDragDistance(event);
|
||
|
if (!event.hasMoved) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var node = Node.getNodeFromTarget(event.target);
|
||
|
|
||
|
if (node) {
|
||
|
if (this.multiselection.start == null) {
|
||
|
this.multiselection.start = node;
|
||
|
}
|
||
|
this.multiselection.end = node;
|
||
|
}
|
||
|
|
||
|
// deselect previous selection
|
||
|
this.deselect();
|
||
|
|
||
|
// find the selected nodes in the range from first to last
|
||
|
var start = this.multiselection.start;
|
||
|
var end = this.multiselection.end || this.multiselection.start;
|
||
|
if (start && end) {
|
||
|
// find the top level childs, all having the same parent
|
||
|
this.multiselection.nodes = this._findTopLevelNodes(start, end);
|
||
|
this.select(this.multiselection.nodes);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* End of multiselect nodes by dragging
|
||
|
* @param event
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onMultiSelectEnd = function (event) {
|
||
|
// set focus to the context menu button of the first node
|
||
|
if (this.multiselection.nodes[0]) {
|
||
|
this.multiselection.nodes[0].dom.menu.focus();
|
||
|
}
|
||
|
|
||
|
this.multiselection.start = null;
|
||
|
this.multiselection.end = null;
|
||
|
|
||
|
// cleanup global event listeners
|
||
|
if (this.mousemove) {
|
||
|
util.removeEventListener(window, 'mousemove', this.mousemove);
|
||
|
delete this.mousemove;
|
||
|
}
|
||
|
if (this.mouseup) {
|
||
|
util.removeEventListener(window, 'mouseup', this.mouseup);
|
||
|
delete this.mouseup;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* deselect currently selected nodes
|
||
|
* @param {boolean} [clearStartAndEnd=false] If true, the `start` and `end`
|
||
|
* state is cleared too.
|
||
|
*/
|
||
|
treemode.deselect = function (clearStartAndEnd) {
|
||
|
this.multiselection.nodes.forEach(function (node) {
|
||
|
node.setSelected(false);
|
||
|
});
|
||
|
this.multiselection.nodes = [];
|
||
|
|
||
|
if (clearStartAndEnd) {
|
||
|
this.multiselection.start = null;
|
||
|
this.multiselection.end = null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* select nodes
|
||
|
* @param {Node[] | Node} nodes
|
||
|
*/
|
||
|
treemode.select = function (nodes) {
|
||
|
if (!Array.isArray(nodes)) {
|
||
|
return this.select([nodes]);
|
||
|
}
|
||
|
|
||
|
if (nodes) {
|
||
|
this.deselect();
|
||
|
|
||
|
this.multiselection.nodes = nodes.slice(0);
|
||
|
|
||
|
var first = nodes[0];
|
||
|
nodes.forEach(function (node) {
|
||
|
node.setSelected(true, node === first);
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* From two arbitrary selected nodes, find their shared parent node.
|
||
|
* From that parent node, select the two child nodes in the brances going to
|
||
|
* nodes `start` and `end`, and select all childs in between.
|
||
|
* @param {Node} start
|
||
|
* @param {Node} end
|
||
|
* @return {Array.<Node>} Returns an ordered list with child nodes
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._findTopLevelNodes = function (start, end) {
|
||
|
var startPath = start.getNodePath();
|
||
|
var endPath = end.getNodePath();
|
||
|
var i = 0;
|
||
|
while (i < startPath.length && startPath[i] === endPath[i]) {
|
||
|
i++;
|
||
|
}
|
||
|
var root = startPath[i - 1];
|
||
|
var startChild = startPath[i];
|
||
|
var endChild = endPath[i];
|
||
|
|
||
|
if (!startChild || !endChild) {
|
||
|
if (root.parent) {
|
||
|
// startChild is a parent of endChild or vice versa
|
||
|
startChild = root;
|
||
|
endChild = root;
|
||
|
root = root.parent
|
||
|
}
|
||
|
else {
|
||
|
// we have selected the root node (which doesn't have a parent)
|
||
|
startChild = root.childs[0];
|
||
|
endChild = root.childs[root.childs.length - 1];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (root && startChild && endChild) {
|
||
|
var startIndex = root.childs.indexOf(startChild);
|
||
|
var endIndex = root.childs.indexOf(endChild);
|
||
|
var firstIndex = Math.min(startIndex, endIndex);
|
||
|
var lastIndex = Math.max(startIndex, endIndex);
|
||
|
|
||
|
return root.childs.slice(firstIndex, lastIndex + 1);
|
||
|
}
|
||
|
else {
|
||
|
return [];
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Event handler for keydown. Handles shortcut keys
|
||
|
* @param {Event} event
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._onKeyDown = function (event) {
|
||
|
var keynum = event.which || event.keyCode;
|
||
|
var ctrlKey = event.ctrlKey;
|
||
|
var shiftKey = event.shiftKey;
|
||
|
var handled = false;
|
||
|
|
||
|
if (keynum == 9) { // Tab or Shift+Tab
|
||
|
var me = this;
|
||
|
setTimeout(function () {
|
||
|
// select all text when moving focus to an editable div
|
||
|
util.selectContentEditable(me.focusTarget);
|
||
|
}, 0);
|
||
|
}
|
||
|
|
||
|
if (this.searchBox) {
|
||
|
if (ctrlKey && keynum == 70) { // Ctrl+F
|
||
|
this.searchBox.dom.search.focus();
|
||
|
this.searchBox.dom.search.select();
|
||
|
handled = true;
|
||
|
}
|
||
|
else if (keynum == 114 || (ctrlKey && keynum == 71)) { // F3 or Ctrl+G
|
||
|
var focus = true;
|
||
|
if (!shiftKey) {
|
||
|
// select next search result (F3 or Ctrl+G)
|
||
|
this.searchBox.next(focus);
|
||
|
}
|
||
|
else {
|
||
|
// select previous search result (Shift+F3 or Ctrl+Shift+G)
|
||
|
this.searchBox.previous(focus);
|
||
|
}
|
||
|
|
||
|
handled = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (this.history) {
|
||
|
if (ctrlKey && !shiftKey && keynum == 90) { // Ctrl+Z
|
||
|
// undo
|
||
|
this._onUndo();
|
||
|
handled = true;
|
||
|
}
|
||
|
else if (ctrlKey && shiftKey && keynum == 90) { // Ctrl+Shift+Z
|
||
|
// redo
|
||
|
this._onRedo();
|
||
|
handled = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (handled) {
|
||
|
event.preventDefault();
|
||
|
event.stopPropagation();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Create main table
|
||
|
* @private
|
||
|
*/
|
||
|
treemode._createTable = function () {
|
||
|
var contentOuter = document.createElement('div');
|
||
|
contentOuter.className = 'jsoneditor-outer';
|
||
|
this.contentOuter = contentOuter;
|
||
|
|
||
|
this.content = document.createElement('div');
|
||
|
this.content.className = 'jsoneditor-tree';
|
||
|
contentOuter.appendChild(this.content);
|
||
|
|
||
|
this.table = document.createElement('table');
|
||
|
this.table.className = 'jsoneditor-tree';
|
||
|
this.content.appendChild(this.table);
|
||
|
|
||
|
// create colgroup where the first two columns don't have a fixed
|
||
|
// width, and the edit columns do have a fixed width
|
||
|
var col;
|
||
|
this.colgroupContent = document.createElement('colgroup');
|
||
|
if (this.options.mode === 'tree') {
|
||
|
col = document.createElement('col');
|
||
|
col.width = "24px";
|
||
|
this.colgroupContent.appendChild(col);
|
||
|
}
|
||
|
col = document.createElement('col');
|
||
|
col.width = "24px";
|
||
|
this.colgroupContent.appendChild(col);
|
||
|
col = document.createElement('col');
|
||
|
this.colgroupContent.appendChild(col);
|
||
|
this.table.appendChild(this.colgroupContent);
|
||
|
|
||
|
this.tbody = document.createElement('tbody');
|
||
|
this.table.appendChild(this.tbody);
|
||
|
|
||
|
this.frame.appendChild(contentOuter);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Show a contextmenu for this node.
|
||
|
* Used for multiselection
|
||
|
* @param {HTMLElement} anchor Anchor element to attache the context menu to.
|
||
|
* @param {function} [onClose] Callback method called when the context menu
|
||
|
* is being closed.
|
||
|
*/
|
||
|
treemode.showContextMenu = function (anchor, onClose) {
|
||
|
var items = [];
|
||
|
var editor = this;
|
||
|
|
||
|
// create duplicate button
|
||
|
items.push({
|
||
|
text: 'Duplicate',
|
||
|
title: 'Duplicate selected fields (Ctrl+D)',
|
||
|
className: 'jsoneditor-duplicate',
|
||
|
click: function () {
|
||
|
Node.onDuplicate(editor.multiselection.nodes);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
// create remove button
|
||
|
items.push({
|
||
|
text: 'Remove',
|
||
|
title: 'Remove selected fields (Ctrl+Del)',
|
||
|
className: 'jsoneditor-remove',
|
||
|
click: function () {
|
||
|
Node.onRemove(editor.multiselection.nodes);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
var menu = new ContextMenu(items, {close: onClose});
|
||
|
menu.show(anchor, this.content);
|
||
|
};
|
||
|
|
||
|
|
||
|
// define modes
|
||
|
module.exports = [
|
||
|
{
|
||
|
mode: 'tree',
|
||
|
mixin: treemode,
|
||
|
data: 'json'
|
||
|
},
|
||
|
{
|
||
|
mode: 'view',
|
||
|
mixin: treemode,
|
||
|
data: 'json'
|
||
|
},
|
||
|
{
|
||
|
mode: 'form',
|
||
|
mixin: treemode,
|
||
|
data: 'json'
|
||
|
}
|
||
|
];
|