2018-06-24 16:07:22 +03:00
|
|
|
/*/* ===========================================================
|
|
|
|
* trumbowyg.history.js v1.0
|
|
|
|
* history plugin for Trumbowyg
|
|
|
|
* http://alex-d.github.com/Trumbowyg
|
|
|
|
* ===========================================================
|
|
|
|
* Author : Sven Dunemann [dunemann@forelabs.eu]
|
|
|
|
*/
|
|
|
|
|
|
|
|
(function ($) {
|
|
|
|
'use strict';
|
|
|
|
$.extend(true, $.trumbowyg, {
|
|
|
|
langs: {
|
2020-05-12 21:47:37 +03:00
|
|
|
// jshint camelcase:false
|
2018-06-24 16:07:22 +03:00
|
|
|
de: {
|
|
|
|
history: {
|
|
|
|
redo: 'Wiederholen',
|
|
|
|
undo: 'Rückgängig'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
en: {
|
|
|
|
history: {
|
|
|
|
redo: 'Redo',
|
|
|
|
undo: 'Undo'
|
|
|
|
}
|
|
|
|
},
|
2020-05-12 21:47:37 +03:00
|
|
|
da: {
|
|
|
|
history: {
|
|
|
|
redo: 'Annuller fortryd',
|
|
|
|
undo: 'Fortryd'
|
|
|
|
}
|
|
|
|
},
|
2018-06-24 16:07:22 +03:00
|
|
|
fr: {
|
|
|
|
history: {
|
|
|
|
redo: 'Annuler',
|
|
|
|
undo: 'Rétablir'
|
|
|
|
}
|
2020-05-12 21:47:37 +03:00
|
|
|
},
|
|
|
|
zh_tw: {
|
|
|
|
history: {
|
|
|
|
redo: '重做',
|
|
|
|
undo: '復原'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
pt_br: {
|
|
|
|
history: {
|
|
|
|
redo: 'Refazer',
|
|
|
|
undo: 'Desfazer'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
ko: {
|
|
|
|
history: {
|
|
|
|
redo: '다시 실행',
|
|
|
|
undo: '되돌리기'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// jshint camelcase:true
|
2018-06-24 16:07:22 +03:00
|
|
|
},
|
|
|
|
plugins: {
|
|
|
|
history: {
|
|
|
|
init: function (t) {
|
|
|
|
t.o.plugins.history = $.extend(true, {
|
|
|
|
_stack: [],
|
|
|
|
_index: -1,
|
|
|
|
_focusEl: undefined
|
|
|
|
}, t.o.plugins.history || {});
|
|
|
|
|
|
|
|
var btnBuildDefRedo = {
|
|
|
|
title: t.lang.history.redo,
|
|
|
|
ico: 'redo',
|
|
|
|
key: 'Y',
|
|
|
|
fn: function () {
|
|
|
|
if (t.o.plugins.history._index < t.o.plugins.history._stack.length - 1) {
|
|
|
|
t.o.plugins.history._index += 1;
|
|
|
|
var index = t.o.plugins.history._index;
|
|
|
|
var newState = t.o.plugins.history._stack[index];
|
|
|
|
|
|
|
|
t.execCmd('html', newState);
|
|
|
|
// because of some semantic optimisations we have to save the state back
|
|
|
|
// to history
|
|
|
|
t.o.plugins.history._stack[index] = t.$ed.html();
|
|
|
|
|
|
|
|
carretToEnd();
|
|
|
|
toggleButtonStates();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var btnBuildDefUndo = {
|
|
|
|
title: t.lang.history.undo,
|
|
|
|
ico: 'undo',
|
|
|
|
key: 'Z',
|
|
|
|
fn: function () {
|
|
|
|
if (t.o.plugins.history._index > 0) {
|
|
|
|
t.o.plugins.history._index -= 1;
|
|
|
|
var index = t.o.plugins.history._index,
|
|
|
|
newState = t.o.plugins.history._stack[index];
|
|
|
|
|
|
|
|
t.execCmd('html', newState);
|
|
|
|
// because of some semantic optimisations we have to save the state back
|
|
|
|
// to history
|
|
|
|
t.o.plugins.history._stack[index] = t.$ed.html();
|
|
|
|
|
|
|
|
carretToEnd();
|
|
|
|
toggleButtonStates();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var pushToHistory = function () {
|
|
|
|
var index = t.o.plugins.history._index,
|
|
|
|
stack = t.o.plugins.history._stack,
|
|
|
|
latestState = stack.slice(-1)[0] || '<p></p>',
|
|
|
|
prevState = stack[index],
|
|
|
|
newState = t.$ed.html(),
|
|
|
|
focusEl = t.doc.getSelection().focusNode,
|
|
|
|
focusElText = '',
|
|
|
|
latestStateTagsList,
|
|
|
|
newStateTagsList,
|
|
|
|
prevFocusEl = t.o.plugins.history._focusEl;
|
|
|
|
|
|
|
|
latestStateTagsList = $('<div>' + latestState + '</div>').find('*').map(function () {
|
|
|
|
return this.localName;
|
|
|
|
});
|
|
|
|
newStateTagsList = $('<div>' + newState + '</div>').find('*').map(function () {
|
|
|
|
return this.localName;
|
|
|
|
});
|
|
|
|
if (focusEl) {
|
|
|
|
t.o.plugins.history._focusEl = focusEl;
|
|
|
|
focusElText = focusEl.outerHTML || focusEl.textContent;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newState !== prevState) {
|
|
|
|
// a new stack entry is defined when current insert ends on a whitespace character
|
|
|
|
// or count of node elements has been changed
|
|
|
|
// or focused element differs from previous one
|
|
|
|
if (focusElText.slice(-1).match(/\s/) ||
|
|
|
|
!arraysAreIdentical(latestStateTagsList, newStateTagsList) ||
|
|
|
|
t.o.plugins.history._index <= 0 || focusEl !== prevFocusEl)
|
|
|
|
{
|
|
|
|
t.o.plugins.history._index += 1;
|
|
|
|
// remove newer entries in history when something new was added
|
|
|
|
// because timeline was changes with interaction
|
|
|
|
t.o.plugins.history._stack = stack.slice(
|
|
|
|
0, t.o.plugins.history._index
|
|
|
|
);
|
2020-05-12 21:47:37 +03:00
|
|
|
// now add new state to modified history
|
2018-06-24 16:07:22 +03:00
|
|
|
t.o.plugins.history._stack.push(newState);
|
|
|
|
} else {
|
|
|
|
// modify last stack entry
|
|
|
|
t.o.plugins.history._stack[index] = newState;
|
|
|
|
}
|
|
|
|
|
|
|
|
toggleButtonStates();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var toggleButtonStates = function () {
|
|
|
|
var index = t.o.plugins.history._index,
|
|
|
|
stackSize = t.o.plugins.history._stack.length,
|
|
|
|
undoState = (index > 0),
|
|
|
|
redoState = (stackSize !== 0 && index !== stackSize - 1);
|
|
|
|
|
|
|
|
toggleButtonState('historyUndo', undoState);
|
|
|
|
toggleButtonState('historyRedo', redoState);
|
|
|
|
};
|
|
|
|
|
|
|
|
var toggleButtonState = function (btn, enable) {
|
|
|
|
var button = t.$box.find('.trumbowyg-' + btn + '-button');
|
|
|
|
|
|
|
|
if (enable) {
|
|
|
|
button.removeClass('trumbowyg-disable');
|
|
|
|
} else if (!button.hasClass('trumbowyg-disable')) {
|
|
|
|
button.addClass('trumbowyg-disable');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var arraysAreIdentical = function (a, b) {
|
|
|
|
if (a === b) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (a == null || b == null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (a.length !== b.length) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var i = 0; i < a.length; i += 1) {
|
|
|
|
if (a[i] !== b[i]) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
var carretToEnd = function () {
|
|
|
|
var node = t.doc.getSelection().focusNode,
|
|
|
|
range = t.doc.createRange();
|
|
|
|
|
|
|
|
if (node.childNodes.length > 0) {
|
|
|
|
range.setStartAfter(node.childNodes[node.childNodes.length - 1]);
|
|
|
|
range.setEndAfter(node.childNodes[node.childNodes.length - 1]);
|
|
|
|
t.doc.getSelection().removeAllRanges();
|
|
|
|
t.doc.getSelection().addRange(range);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
t.$c.on('tbwinit tbwchange', pushToHistory);
|
|
|
|
|
|
|
|
t.addBtnDef('historyRedo', btnBuildDefRedo);
|
|
|
|
t.addBtnDef('historyUndo', btnBuildDefUndo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
})(jQuery);
|