cryptpad/www/kanban/inner.js

1347 lines
48 KiB
JavaScript
Raw Normal View History

2018-04-01 16:52:21 +00:00
define([
'jquery',
2018-05-22 16:22:25 +00:00
'json.sortify',
2018-04-01 16:52:21 +00:00
'/bower_components/nthen/index.js',
'/common/sframe-common.js',
'/common/sframe-app-framework.js',
2020-03-03 16:40:01 +00:00
'/common/sframe-common-codemirror.js',
'/common/common-util.js',
'/common/common-hash.js',
2018-05-16 10:22:43 +00:00
'/common/common-interface.js',
2020-03-09 17:49:57 +00:00
'/common/common-ui-elements.js',
'/common/inner/common-mediatag.js',
2018-04-01 16:52:21 +00:00
'/customize/messages.js',
2020-03-03 15:20:45 +00:00
'/common/hyperscript.js',
2020-03-03 16:40:01 +00:00
'/common/text-cursor.js',
2020-03-04 16:48:20 +00:00
'/common/diffMarked.js',
2020-03-03 16:40:01 +00:00
'/bower_components/chainpad/chainpad.dist.js',
2020-03-03 15:20:45 +00:00
'/bower_components/marked/marked.min.js',
'cm/lib/codemirror',
'/kanban/jkanban_cp.js',
2021-06-15 12:48:58 +00:00
'/kanban/export.js',
2020-03-03 15:20:45 +00:00
'cm/mode/gfm/gfm',
'cm/addon/edit/closebrackets',
'cm/addon/edit/matchbrackets',
'cm/addon/edit/trailingspace',
'cm/addon/selection/active-line',
'cm/addon/search/search',
'cm/addon/search/match-highlighter',
2020-03-03 15:20:45 +00:00
'css!/bower_components/codemirror/lib/codemirror.css',
'css!/bower_components/codemirror/addon/dialog/dialog.css',
'css!/bower_components/codemirror/addon/fold/foldgutter.css',
2018-07-14 13:15:23 +00:00
'less!/kanban/app-kanban.less'
2018-04-01 16:52:21 +00:00
], function (
$,
2018-05-22 16:22:25 +00:00
Sortify,
2018-04-01 16:52:21 +00:00
nThen,
SFCommon,
Framework,
2020-03-03 16:40:01 +00:00
SFCodeMirror,
Util,
Hash,
2018-05-16 10:22:43 +00:00
UI,
2020-03-09 17:49:57 +00:00
UIElements,
MT,
2020-03-03 15:20:45 +00:00
Messages,
h,
2020-03-03 16:40:01 +00:00
TextCursor,
2020-03-04 16:48:20 +00:00
DiffMd,
2020-03-03 16:40:01 +00:00
ChainPad,
2020-03-03 15:20:45 +00:00
Marked,
CodeMirror,
2021-06-15 12:48:58 +00:00
jKanban,
Export)
2018-05-16 10:22:43 +00:00
{
2018-05-17 08:27:16 +00:00
var verbose = function (x) { console.log(x); };
verbose = function () {}; // comment out to enable verbose logging
2020-03-05 11:06:35 +00:00
var onRedraw = Util.mkEvent();
2020-03-09 17:49:57 +00:00
var onCursorUpdate = Util.mkEvent();
var remoteCursors = {};
2018-05-17 08:27:16 +00:00
2020-03-03 17:35:56 +00:00
var setValueAndCursor = function (input, val, _cursor) {
if (!input) { return; }
var $input = $(input);
var focus = _cursor || $input.is(':focus');
var oldVal = $input.val();
2020-03-04 16:48:20 +00:00
var ops = ChainPad.Diff.diff(_cursor ? _cursor.value : oldVal, val);
2020-03-03 17:35:56 +00:00
var cursor = _cursor || input;
var selects = ['selectionStart', 'selectionEnd'].map(function (attr) {
return TextCursor.transformCursor(cursor[attr], ops);
});
$input.val(val);
if (focus) { $input.focus(); }
input.selectionStart = selects[0];
input.selectionEnd = selects[1];
};
2020-03-03 16:40:01 +00:00
2020-03-09 17:49:57 +00:00
var getTextColor = function (hex) {
if (hex && /^#/.test(hex)) { hex = hex.slice(1); }
if (!/^[0-9a-f]{6}$/i.test(hex)) {
return '#000000';
}
var r = parseInt(hex.slice(0,2), 16);
var g = parseInt(hex.slice(2,4), 16);
var b = parseInt(hex.slice(4,6), 16);
if ((r*0.213 + g*0.715 + b*0.072) > 255/2) {
return '#000000';
}
return '#FFFFFF';
};
var getAvatar = function (cursor, noClear) {
// Tippy
var html = MT.getCursorAvatar(cursor);
2020-03-09 17:49:57 +00:00
2021-08-26 11:34:38 +00:00
var name = UI.getDisplayName(cursor.name);
var l; // label?
2021-08-26 11:34:38 +00:00
var animal = '';
if (cursor.name === Messages.anonymous && typeof(cursor.uid) === 'string') {
l = MT.getPseudorandomAnimal(cursor.uid);
if (l) {
animal = '.animal';
}
}
if (!l) {
2021-08-26 11:34:38 +00:00
l = MT.getPrettyInitials(name);
}
2020-03-09 17:49:57 +00:00
var text = '';
if (cursor.color) {
2021-08-26 11:34:38 +00:00
text = 'background-color:' + cursor.color + '; color:'+getTextColor(cursor.color)+';';
2020-03-09 17:49:57 +00:00
}
2021-08-26 11:34:38 +00:00
var avatar = h('span.cp-cursor.cp-tippy-html' + animal, {
style: text,
2020-04-29 10:30:59 +00:00
'data-cptippy-html': true,
2021-08-26 11:34:38 +00:00
title: html,
2020-03-09 17:49:57 +00:00
}, l);
if (!noClear) {
cursor.clear = function () {
$(avatar).remove();
};
}
return avatar;
};
2020-03-04 17:50:17 +00:00
var getExistingTags = function (boards) {
var tags = [];
boards = boards || {};
Object.keys(boards.items || {}).forEach(function (id) {
var data = boards.items[id];
if (!Array.isArray(data.tags)) { return; }
2020-03-10 14:51:39 +00:00
data.tags.forEach(function (_tag) {
var tag = _tag.toLowerCase();
2020-03-04 17:50:17 +00:00
if (tags.indexOf(tag) === -1) { tags.push(tag); }
});
});
tags.sort();
return tags;
};
2020-03-03 16:40:01 +00:00
var addEditItemButton = function () {};
2021-02-03 11:14:05 +00:00
var now = function () { return +new Date(); };
var _lastUpdate = 0;
var _updateBoards = function (framework, kanban, boards) {
_lastUpdate = now();
kanban.setBoards(Util.clone(boards));
kanban.inEditMode = false;
addEditItemButton(framework, kanban);
2021-02-03 11:14:05 +00:00
};
2021-02-03 11:52:54 +00:00
var _updateBoardsThrottle = Util.throttle(_updateBoards, 1000);
2021-02-03 11:14:05 +00:00
var updateBoards = function (framework, kanban, boards) {
if ((now() - _lastUpdate) > 5000 || framework.isLocked()) {
2021-02-03 11:14:05 +00:00
_updateBoards(framework, kanban, boards);
2021-02-03 11:14:54 +00:00
return;
2021-02-03 11:14:05 +00:00
}
2021-02-03 11:41:22 +00:00
_updateBoardsThrottle(framework, kanban, boards);
2021-02-03 11:14:05 +00:00
};
2020-03-03 16:40:01 +00:00
var onRemoteChange = Util.mkEvent();
2020-03-03 15:20:45 +00:00
var editModal;
var PROPERTIES = ['title', 'body', 'tags', 'color'];
var BOARD_PROPERTIES = ['title', 'color'];
var createEditModal = function (framework, kanban) {
if (framework.isReadOnly()) { return; }
if (editModal) { return editModal; }
2020-03-03 16:40:01 +00:00
var dataObject = {};
var isBoard, id;
var offline = false;
2020-03-03 16:40:01 +00:00
var update = function () {
updateBoards(framework, kanban, kanban.options.boards);
};
var commit = function () {
framework.localChange();
update();
2020-03-03 16:40:01 +00:00
};
2020-03-09 17:49:57 +00:00
var conflicts, conflictContainer, titleInput, tagsDiv, colors, text;
2020-03-03 15:20:45 +00:00
var content = h('div', [
2020-03-09 17:49:57 +00:00
conflictContainer = h('div#cp-kanban-edit-conflicts', [
h('div', Messages.kanban_conflicts),
conflicts = h('div.cp-kanban-cursors')
]),
2020-03-03 15:20:45 +00:00
h('label', {for:'cp-kanban-edit-title'}, Messages.kanban_title),
titleInput = h('input#cp-kanban-edit-title'),
h('label', {for:'cp-kanban-edit-body'}, Messages.kanban_body),
h('div#cp-kanban-edit-body', [
text = h('textarea')
]),
h('label', {for:'cp-kanban-edit-tags'}, Messages.fm_tagsName),
tagsDiv = h('div#cp-kanban-edit-tags'),
h('label', {for:'cp-kanban-edit-color'}, Messages.kanban_color),
colors = h('div#cp-kanban-edit-colors'),
]);
var $tags = $(tagsDiv);
2020-03-03 15:20:45 +00:00
2020-03-09 17:49:57 +00:00
var $conflict = $(conflicts);
var $cc = $(conflictContainer);
var conflict = {
setValue: function () {
$conflict.empty();
var i = 0;
$cc.hide();
Object.keys(remoteCursors).forEach(function (nid) {
var c = remoteCursors[nid];
var avatar = getAvatar(c, true);
if (Number(c.item) === Number(id) || Number(c.board) === Number(id)) {
$conflict.append(avatar);
i++;
}
});
if (!i) { return; }
$cc.show();
}
};
2020-03-03 15:20:45 +00:00
// Title
var $title = $(titleInput);
2020-03-03 16:40:01 +00:00
$title.on('change keyup', function () {
dataObject.title = $title.val();
commit();
});
2020-03-03 15:20:45 +00:00
var title = {
getValue: function () {
return $title.val();
},
2020-03-03 16:40:01 +00:00
setValue: function (val, preserveCursor) {
if (!preserveCursor) {
$title.val(val);
} else {
2020-03-03 17:35:56 +00:00
setValueAndCursor(titleInput, val);
2020-03-03 16:40:01 +00:00
}
2020-03-03 15:20:45 +00:00
}
};
// Body
var cm = SFCodeMirror.create("gfm", CodeMirror, text);
var editor = cm.editor;
editor.setOption('gutters', []);
editor.setOption('lineNumbers', false);
editor.setOption('readOnly', false);
editor.on('keydown', function (editor, e) {
if (e.which === 27) {
// Focus the next form element but don't close the modal (stopPropagation)
$tags.find('.token-input').focus();
}
e.stopPropagation();
});
2020-03-03 15:20:45 +00:00
var common = framework._.sfCommon;
var markdownTb = common.createMarkdownToolbar(editor, {
embed: function (mt) {
editor.focus();
editor.replaceSelection($(mt)[0].outerHTML);
}
});
2020-03-03 15:20:45 +00:00
$(text).before(markdownTb.toolbar);
$(markdownTb.toolbar).show();
editor.refresh();
var body = {
getValue: function () {
return editor.getValue();
},
2020-03-03 16:40:01 +00:00
setValue: function (val, preserveCursor) {
if (isBoard) { return; }
if (!preserveCursor) {
editor.setValue(val || '');
editor.save();
2020-03-03 16:40:01 +00:00
} else {
SFCodeMirror.setValueAndCursor(editor, editor.getValue(), val || '');
}
},
refresh: function () {
editor.refresh();
2020-03-03 15:20:45 +00:00
}
};
cm.configureTheme(common, function () {});
2020-03-11 15:57:56 +00:00
SFCodeMirror.mkIndentSettings(editor, framework._.cpNfInner.metadataMgr);
2020-03-03 16:40:01 +00:00
editor.on('change', function () {
var val = editor.getValue();
if (dataObject.body === val) { return; }
dataObject.body = val;
2020-03-03 16:40:01 +00:00
commit();
});
2020-03-03 15:20:45 +00:00
setTimeout(function () {
var privateData = framework._.cpNfInner.metadataMgr.getPrivateData();
var fmConfig = {
dropArea: $('.CodeMirror'),
body: $('body'),
onUploaded: function (ev, data) {
var parsed = Hash.parsePadUrl(data.url);
var secret = Hash.getSecrets('file', parsed.hash, data.password);
var fileHost = privateData.fileHost || privateData.origin;
var src = fileHost + Hash.getBlobPathFromHex(secret.channel);
var key = Hash.encodeBase64(secret.keys.cryptKey);
var mt = UI.mediaTag(src, key).outerHTML;
editor.replaceSelection(mt);
}
};
common.createFileManager(fmConfig);
});
2020-03-03 15:20:45 +00:00
// Tags
2020-03-03 16:40:01 +00:00
var _field, initialTags;
2020-03-03 15:20:45 +00:00
var tags = {
getValue: function () {
if (!_field) { return; }
return _field.getTokens();
},
2020-03-03 16:40:01 +00:00
setValue: function (tags, preserveCursor) {
if (isBoard) { return; }
if (preserveCursor && initialTags && Sortify(tags || []) === initialTags) {
// Don't redraw if there is no change
return;
}
initialTags = Sortify(tags || []);
2020-03-03 15:20:45 +00:00
$tags.empty();
var input = UI.dialog.textInput();
$tags.append(input);
2020-03-04 17:50:17 +00:00
var existing = getExistingTags(kanban.options.boards);
2020-03-03 15:20:45 +00:00
_field = UI.tokenField(input, existing).preventDuplicates(function (val) {
UI.warn(Messages._getKey('tags_duplicate', [val]));
});
2020-03-03 16:40:01 +00:00
_field.setTokens(tags || []);
$tags.find('.token-input').on('keydown', function (e) {
// if the tokenfield is blank and the user hits enter or escape
// then allow the event to propogate (closing the modal)
// this can leave behind the autocomplete menu, so forcefully hide it
if (!$(this).val() && [13, 27].indexOf(e.which) !== -1) {
return void $('.ui-autocomplete.ui-front').hide();
}
e.stopPropagation();
});
2020-03-03 16:40:01 +00:00
var commitTags = function () {
if (offline) { return; }
2020-03-11 12:06:18 +00:00
setTimeout(function () {
dataObject.tags = Util.deduplicateString(_field.getTokens().map(function (t) {
return t.toLowerCase();
}));
initialTags = Sortify(dataObject.tags);
commit();
});
2020-03-03 16:40:01 +00:00
};
_field.tokenfield.on('tokenfield:createdtoken', commitTags);
_field.tokenfield.on('tokenfield:editedoken', commitTags);
_field.tokenfield.on('tokenfield:removedtoken', commitTags);
2020-03-03 15:20:45 +00:00
}
2020-03-03 16:42:08 +00:00
};
2020-03-03 15:20:45 +00:00
// Colors
var $colors = $(colors);
var palette = [''];
for (var i=1; i<=8; i++) { palette.push('color'+i); }
var selectedColor = '';
palette.forEach(function (color) {
var $color = $(h('span.cp-kanban-palette.fa'));
$color.addClass('cp-kanban-palette-'+(color || 'nocolor'));
$color.click(function () {
if (offline) { return; }
2020-03-03 16:40:01 +00:00
if (color === selectedColor) { return; }
2020-03-03 15:20:45 +00:00
selectedColor = color;
$colors.find('.cp-kanban-palette').removeClass('fa-check');
var $col = $colors.find('.cp-kanban-palette-'+(color || 'nocolor'));
$col.addClass('fa-check');
2020-03-03 16:40:01 +00:00
dataObject.color = color;
commit();
2020-03-03 15:20:45 +00:00
}).appendTo($colors);
});
var color = {
getValue: function () {
return selectedColor;
},
setValue: function (color) {
$colors.find('.cp-kanban-palette').removeClass('fa-check');
var $col = $colors.find('.cp-kanban-palette-'+(color || 'nocolor'));
$col.addClass('fa-check');
2020-03-09 13:58:49 +00:00
selectedColor = color;
2020-03-03 15:20:45 +00:00
}
};
var button = [{
2020-03-05 11:06:35 +00:00
className: 'danger left',
2020-03-03 15:20:45 +00:00
name: Messages.kanban_delete,
2020-03-13 14:05:07 +00:00
confirm: true,
2020-03-13 14:40:03 +00:00
onClick: function (/*button*/) {
2020-03-03 16:40:01 +00:00
var boards = kanban.options.boards || {};
if (isBoard) {
var list = boards.list || [];
var idx = list.indexOf(id);
if (idx !== -1) { list.splice(idx, 1); }
delete (boards.data || {})[id];
2020-06-02 08:54:15 +00:00
kanban.removeBoard(id);
2020-03-03 16:40:01 +00:00
return void commit();
}
Object.keys(boards.data || {}).forEach(function (boardId) {
var board = boards.data[boardId];
if (!board) { return; }
var items = board.item || [];
var idx = items.indexOf(id);
if (idx !== -1) { items.splice(idx, 1); }
});
delete (boards.items || {})[id];
commit();
2020-03-03 15:20:45 +00:00
},
keys: []
}, {
className: 'primary',
2020-03-03 16:40:01 +00:00
name: Messages.filePicker_close,
2020-03-03 15:20:45 +00:00
onClick: function () {
2020-03-09 17:49:57 +00:00
onCursorUpdate.fire({});
2020-03-03 15:20:45 +00:00
},
keys: [13, 27]
2020-03-03 15:20:45 +00:00
}];
var modal = UI.dialog.customModal(content, {
buttons: button
});
2020-03-09 13:58:49 +00:00
modal.classList.add('cp-kanban-edit-modal');
var $modal = $(modal);
framework.onEditableChange(function (unlocked) {
editor.setOption('readOnly', !unlocked);
$title.prop('disabled', unlocked ? '' : 'disabled');
2021-02-04 16:26:18 +00:00
if (_field) {
$(_field.element).tokenfield(unlocked ? 'enable' : 'disable');
}
$modal.find('nav button.danger').prop('disabled', unlocked ? '' : 'disabled');
offline = !unlocked;
});
2020-03-03 16:40:01 +00:00
2020-03-13 14:05:07 +00:00
var setId = function (_isBoard, _id) {
// Reset the mdoal with a new id
isBoard = _isBoard;
2020-05-25 12:16:37 +00:00
id = Number(_id);
2020-03-13 14:05:07 +00:00
if (_isBoard) {
onCursorUpdate.fire({
board: _id
});
dataObject = kanban.getBoardJSON(id);
$(content)
.find('#cp-kanban-edit-body, #cp-kanban-edit-tags, [for="cp-kanban-edit-body"], [for="cp-kanban-edit-tags"]')
.hide();
} else {
onCursorUpdate.fire({
item: _id
});
dataObject = kanban.getItemJSON(id);
$(content)
.find('#cp-kanban-edit-body, #cp-kanban-edit-tags, [for="cp-kanban-edit-body"], [for="cp-kanban-edit-tags"]')
.show();
}
// Also reset the buttons
$modal.find('nav').after(UI.dialog.getButtons(button)).remove();
2020-03-13 14:05:07 +00:00
};
2020-03-03 16:40:01 +00:00
onRemoteChange.reg(function () {
if (isBoard) {
2020-03-03 17:35:56 +00:00
dataObject = kanban.getBoardJSON(id);
2020-03-03 16:40:01 +00:00
} else {
2020-03-03 17:35:56 +00:00
dataObject = kanban.getItemJSON(id);
2020-03-03 16:40:01 +00:00
}
2020-03-10 16:12:41 +00:00
// Check if our item has been deleted
2020-03-03 16:40:01 +00:00
if (!dataObject) {
var $frame = $(modal).parents('.alertify').first();
if ($frame[0] && $frame[0].closeModal) {
$frame[0].closeModal();
}
return;
}
// Not deleted, apply updates
2020-03-09 17:49:57 +00:00
editModal.conflict.setValue();
2020-03-03 16:40:01 +00:00
PROPERTIES.forEach(function (type) {
editModal[type].setValue(dataObject[type], true);
});
});
2020-03-03 15:20:45 +00:00
return {
modal: modal,
setId: setId,
title: title,
body: body,
tags: tags,
2020-03-09 17:49:57 +00:00
color: color,
conflict: conflict
2020-03-03 15:20:45 +00:00
};
};
var getItemEditModal = function (framework, kanban, eid) {
// Create modal if needed
if (!editModal) { editModal = createEditModal(framework, kanban); }
editModal.setId(false, eid);
var boards = kanban.options.boards || {};
var item = (boards.items || {})[eid];
if (!item) { return void UI.warn(Messages.error); }
2020-03-09 17:49:57 +00:00
editModal.conflict.setValue();
2020-03-03 15:20:45 +00:00
PROPERTIES.forEach(function (type) {
if (!editModal[type]) { return; }
editModal[type].setValue(item[type]);
});
UI.openCustomModal(editModal.modal);
editModal.body.refresh();
2020-03-03 15:20:45 +00:00
};
var getBoardEditModal = function (framework, kanban, id) {
// Create modal if needed
if (!editModal) { editModal = createEditModal(framework, kanban); }
editModal.setId(true, id);
var boards = kanban.options.boards || {};
var board = (boards.data || {})[id];
if (!board) { return void UI.warn(Messages.error); }
2020-03-09 17:49:57 +00:00
editModal.conflict.setValue();
2020-03-03 15:20:45 +00:00
BOARD_PROPERTIES.forEach(function (type) {
if (!editModal[type]) { return; }
editModal[type].setValue(board[type]);
});
UI.openCustomModal(editModal.modal);
};
2020-03-03 16:40:01 +00:00
addEditItemButton = function (framework, kanban) {
2018-05-25 09:49:43 +00:00
if (!kanban) { return; }
if (framework.isReadOnly() || framework.isLocked()) { return; }
2018-05-22 16:22:25 +00:00
var $container = $(kanban.element);
2020-03-02 11:05:09 +00:00
$container.find('.kanban-edit-item').remove();
$container.find('.kanban-item').each(function (i, el) {
var itemId = $(el).attr('data-eid');
2018-05-23 12:24:18 +00:00
$('<button>', {
2020-06-19 15:45:01 +00:00
'class': 'kanban-edit-item fa fa-pencil',
'alt': Messages.kanban_editCard,
2018-05-23 12:24:18 +00:00
}).click(function (e) {
2020-03-03 15:20:45 +00:00
getItemEditModal(framework, kanban, itemId);
2018-05-22 16:22:25 +00:00
e.stopPropagation();
2020-03-04 16:48:20 +00:00
}).insertAfter($(el).find('.kanban-item-text'));
2018-05-22 16:22:25 +00:00
});
2020-03-02 11:05:09 +00:00
$container.find('.kanban-board').each(function (i, el) {
var itemId = $(el).attr('data-id');
$('<button>', {
2020-06-19 15:45:01 +00:00
'class': 'kanban-edit-item fa fa-pencil',
'alt': Messages.kanban_editBoard,
2020-03-02 11:05:09 +00:00
}).click(function (e) {
2020-03-03 15:20:45 +00:00
getBoardEditModal(framework, kanban, itemId);
2020-03-02 11:05:09 +00:00
e.stopPropagation();
}).appendTo($(el).find('.kanban-board-header'));
});
2018-05-22 16:22:25 +00:00
};
// Kanban code
2020-03-05 12:49:58 +00:00
var getDefaultBoards = function () {
var items = {};
for (var i=1; i<=6; i++) {
2020-03-02 14:03:32 +00:00
items[i] = {
id: i,
title: Messages._getKey('kanban_item', [i])
};
}
var defaultBoards = {
2020-03-02 14:03:32 +00:00
list: [11, 12, 13],
data: {
2020-03-02 14:03:32 +00:00
"11": {
"id": 11,
"title": Messages.kanban_todo,
2020-03-02 14:03:32 +00:00
"item": [1, 2]
},
2020-03-02 14:03:32 +00:00
"12": {
"id": 12,
"title": Messages.kanban_working,
"item": [],
},
2020-03-02 14:03:32 +00:00
"13": {
"id": 13,
"title": Messages.kanban_done,
"item": [],
}
},
items: items
};
2020-03-05 12:49:58 +00:00
return defaultBoards;
};
var migrate = function (framework, boards) {
if (!Array.isArray(boards)) { return; }
console.log("Migration to new format");
var b = {
list: [],
data: {},
items: {}
};
var i = 1;
boards.forEach(function (board) {
board.id = i;
b.list.push(i);
b.data[i] = board;
i++;
if (!Array.isArray(board.item)) { return; }
board.item = board.item.map(function (item) {
item.id = i;
b.items[i] = item;
return i++; // return current id and incrmeent after
});
});
return b;
};
2018-04-01 16:52:21 +00:00
2020-03-05 12:49:58 +00:00
var initKanban = function (framework, boards) {
2020-03-05 13:07:34 +00:00
var migrated = false;
2018-05-16 10:22:43 +00:00
if (!boards) {
2018-05-17 08:27:16 +00:00
verbose("Initializing with default boards content");
2020-03-05 12:49:58 +00:00
boards = getDefaultBoards();
} else if (Array.isArray(boards)) {
2020-03-05 12:49:58 +00:00
boards = migrate(framework, boards);
migrated = true;
} else {
2018-05-17 08:27:16 +00:00
verbose("Initializing with boards content " + boards);
}
// Remove any existing elements
$(".kanban-container-outer").remove();
2018-05-16 17:28:30 +00:00
var getInput = function () {
return $('<input>', {
'type': 'text',
'id': 'kanban-edit',
'size': '30'
2018-05-30 13:00:26 +00:00
}).click(function (e) { e.stopPropagation(); });
2018-05-16 17:28:30 +00:00
};
2020-03-10 16:38:49 +00:00
var openLink = function (href) {
if (/^\/[^\/]/.test(href)) {
var privateData = framework._.cpNfInner.metadataMgr.getPrivateData();
href = privateData.origin + href;
}
framework._.sfCommon.openUnsafeURL(href);
};
var md = framework._.cpNfInner.metadataMgr.getPrivateData();
var _tagsAnd = Util.find(md, ['settings', 'kanban', 'tagsAnd']);
2020-03-10 16:38:49 +00:00
var kanban = new jKanban({
element: '#cp-app-kanban-content',
gutter: '5px',
widthBoard: '300px',
buttonContent: '❌',
2021-02-08 10:06:32 +00:00
readOnly: framework.isReadOnly() || framework.isLocked(),
tagsAnd: _tagsAnd,
2020-03-05 11:06:35 +00:00
refresh: function () {
onRedraw.fire();
},
onChange: function () {
2018-05-17 08:27:16 +00:00
verbose("Board object has changed");
framework.localChange();
2018-05-22 16:22:25 +00:00
if (kanban) {
2020-03-02 11:05:09 +00:00
addEditItemButton(framework, kanban);
2018-05-22 16:22:25 +00:00
}
},
click: function (el) {
if (framework.isReadOnly() || framework.isLocked()) { return; }
if (kanban.inEditMode) {
2018-05-30 13:00:26 +00:00
$(el).focus();
2018-05-17 08:27:16 +00:00
verbose("An edit is already active");
2018-05-30 13:00:26 +00:00
//return;
}
var eid = $(el).attr('data-eid');
kanban.inEditMode = eid;
2020-03-09 17:49:57 +00:00
setTimeout(function () {
// Make sure the click is sent after the "blur" in case we move from a card to another
onCursorUpdate.fire({
item: eid
});
});
var name = $(el).text();
$(el).html('');
// Add input
2018-05-16 17:28:30 +00:00
var $input = getInput().val(name).appendTo(el).focus();
$input[0].select();
2018-05-16 17:28:30 +00:00
var save = function () {
// Store the value
var name = $input.val();
// Remove the input
$(el).text(name);
2018-05-16 17:28:30 +00:00
// Save the value for the correct board
var item = kanban.getItemJSON(eid);
item.title = name;
kanban.onChange();
2018-05-16 17:28:30 +00:00
// Unlock edit mode
kanban.inEditMode = false;
2020-03-09 17:49:57 +00:00
onCursorUpdate.fire({});
2018-05-16 17:28:30 +00:00
};
2020-03-10 15:51:30 +00:00
$input.blur(save);
2018-05-16 17:28:30 +00:00
$input.keydown(function (e) {
if (e.which === 13) {
e.preventDefault();
e.stopPropagation();
save();
if (!$input.val()) { return; }
if (!$(el).closest('.kanban-item').is(':last-child')) { return; }
2020-03-10 16:05:17 +00:00
$(el).closest('.kanban-board').find('.kanban-title-button').click();
2018-05-16 17:28:30 +00:00
return;
}
if (e.which === 27) {
e.preventDefault();
e.stopPropagation();
2020-03-09 17:49:57 +00:00
save();
2018-05-16 17:28:30 +00:00
}
});
2020-03-03 17:35:56 +00:00
$input.on('change keyup', function () {
var item = kanban.getItemJSON(eid);
if (!item) { return; }
var name = $input.val();
item.title = name;
framework.localChange();
});
2018-04-01 16:52:21 +00:00
},
2018-05-16 17:28:30 +00:00
boardTitleClick: function (el, e) {
e.stopPropagation();
if (framework.isReadOnly() || framework.isLocked()) { return; }
if (kanban.inEditMode) {
2018-05-30 13:00:26 +00:00
$(el).focus();
2018-05-17 08:27:16 +00:00
verbose("An edit is already active");
2018-05-30 13:00:26 +00:00
//return;
}
var boardId = $(el).closest('.kanban-board').attr("data-id");
kanban.inEditMode = boardId;
2020-03-09 17:49:57 +00:00
setTimeout(function () {
// Make sure the click is sent after the "blur" in case we move from a card to another
onCursorUpdate.fire({
board: boardId
});
});
var name = $(el).text();
$(el).html('');
2018-05-16 17:28:30 +00:00
var $input = getInput().val(name).appendTo(el).focus();
$input[0].select();
2018-05-16 17:28:30 +00:00
var save = function () {
// Store the value
var name = $input.val();
2019-12-12 15:09:07 +00:00
if (!name || !name.trim()) {
return kanban.onChange();
}
2018-05-16 17:28:30 +00:00
// Remove the input
$(el).text(name);
2018-05-16 17:28:30 +00:00
// Save the value for the correct board
kanban.getBoardJSON(boardId).title = name;
kanban.onChange();
2018-05-16 17:28:30 +00:00
// Unlock edit mode
kanban.inEditMode = false;
2020-03-09 17:49:57 +00:00
onCursorUpdate.fire({});
2018-05-16 17:28:30 +00:00
};
$input.blur(save);
$input.keydown(function (e) {
if (e.which === 13) {
e.preventDefault();
e.stopPropagation();
save();
return;
}
if (e.which === 27) {
e.preventDefault();
e.stopPropagation();
2020-03-09 17:49:57 +00:00
save();
2018-05-16 17:28:30 +00:00
return;
}
});
2020-03-03 17:35:56 +00:00
$input.on('change keyup', function () {
var item = kanban.getBoardJSON(boardId);
if (!item) { return; }
var name = $input.val();
item.title = name;
framework.localChange();
});
},
2018-05-22 16:22:25 +00:00
addItemClick: function (el) {
if (framework.isReadOnly() || framework.isLocked()) { return; }
2020-06-03 11:16:10 +00:00
var $el = $(el);
2018-05-16 17:28:30 +00:00
if (kanban.inEditMode) {
2020-06-03 11:16:10 +00:00
$el.focus();
2018-05-17 08:27:16 +00:00
verbose("An edit is already active");
2018-05-30 13:00:26 +00:00
//return;
2018-05-16 17:28:30 +00:00
}
kanban.inEditMode = "new";
// create a form to enter element
2020-06-03 11:16:10 +00:00
var isTop = $el.attr('data-top');
var boardId = $el.closest('.kanban-board').attr("data-id");
2019-12-09 10:03:11 +00:00
var $item = $('<div>', {'class': 'kanban-item new-item'});
if (isTop) {
$item.addClass('item-top');
}
2018-05-16 17:28:30 +00:00
var $input = getInput().val(name).appendTo($item);
2020-06-03 11:16:10 +00:00
kanban.addForm(boardId, $item[0], isTop);
2018-05-16 17:28:30 +00:00
$input.focus();
2020-03-04 14:25:03 +00:00
setTimeout(function () {
2020-06-03 11:16:10 +00:00
if (isTop) {
$el.closest('.kanban-drag').scrollTop(0);
} else {
$input[0].scrollIntoView();
}
2020-03-04 14:25:03 +00:00
});
2018-05-16 17:28:30 +00:00
var save = function () {
$item.remove();
2018-05-22 16:22:25 +00:00
kanban.inEditMode = false;
2020-03-09 17:49:57 +00:00
onCursorUpdate.fire({});
2018-05-22 16:22:25 +00:00
if (!$input.val()) { return; }
2020-03-02 14:03:32 +00:00
var id = Util.createRandomInteger();
2020-03-16 13:49:59 +00:00
while (kanban.getItemJSON(id)) {
id = Util.createRandomInteger();
}
2020-03-05 11:06:35 +00:00
var item = {
"id": id,
2018-05-16 17:28:30 +00:00
"title": $input.val(),
2020-03-05 11:06:35 +00:00
};
if (kanban.options.tags && kanban.options.tags.length) {
item.tags = kanban.options.tags;
}
2020-06-03 11:16:10 +00:00
kanban.addElement(boardId, item, isTop);
2018-05-16 10:22:43 +00:00
};
2018-05-16 17:28:30 +00:00
$input.blur(save);
$input.keydown(function (e) {
if (e.which === 13) {
e.preventDefault();
e.stopPropagation();
save();
if (!$input.val()) { return; }
2020-06-03 11:16:10 +00:00
var $footer = $el.closest('.kanban-board').find('footer');
if (isTop) {
$footer.find('.kanban-title-button[data-top]').click();
} else {
$footer.find('.kanban-title-button').click();
}
2018-05-16 17:28:30 +00:00
return;
}
if (e.which === 27) {
e.preventDefault();
e.stopPropagation();
$item.remove();
2018-05-16 17:28:30 +00:00
kanban.inEditMode = false;
2020-03-09 17:49:57 +00:00
onCursorUpdate.fire({});
2018-05-16 17:28:30 +00:00
return;
}
});
},
applyHtml: function (html, node) {
DiffMd.apply(html, $(node),framework._.sfCommon);
},
2020-03-04 16:48:20 +00:00
renderMd: function (md) {
return DiffMd.render(md);
2020-03-04 16:48:20 +00:00
},
addItemButton: true,
2020-03-09 17:49:57 +00:00
getTextColor: getTextColor,
getAvatar: getAvatar,
2020-03-10 16:38:49 +00:00
openLink: openLink,
getTags: getExistingTags,
2020-03-09 17:49:57 +00:00
cursors: remoteCursors,
boards: boards,
_boards: Util.clone(boards),
});
2018-04-01 16:52:21 +00:00
framework._.cpNfInner.metadataMgr.onChange(function () {
var md = framework._.cpNfInner.metadataMgr.getPrivateData();
var tagsAnd = Util.find(md, ['settings', 'kanban', 'tagsAnd']);
if (_tagsAnd === tagsAnd) { return; }
// If the rendering has changed, update the value and redraw
kanban.options.tagsAnd = tagsAnd;
_tagsAnd = tagsAnd;
2021-02-03 11:52:54 +00:00
updateBoards(framework, kanban, kanban.options.boards);
});
2020-03-05 12:49:58 +00:00
if (migrated) { framework.localChange(); }
var addBoardDefault = document.getElementById('kanban-addboard');
2018-05-23 12:24:18 +00:00
$(addBoardDefault).attr('title', Messages.kanban_addBoard);
addBoardDefault.addEventListener('click', function () {
2021-02-08 10:06:32 +00:00
if (framework.isReadOnly() || framework.isLocked()) { return; }
/*var counter = 1;
2018-05-16 10:22:43 +00:00
// Get the new board id
2018-05-22 16:22:25 +00:00
var boardExists = function (b) { return b.id === "board" + counter; };
while (kanban.options.boards.some(boardExists)) { counter++; }
*/
2020-03-02 14:03:32 +00:00
var id = Util.createRandomInteger();
2020-03-16 13:49:59 +00:00
while (kanban.getBoardJSON(id)) {
id = Util.createRandomInteger();
}
2018-04-01 16:52:21 +00:00
kanban.addBoard({
"id": id,
2018-05-16 14:45:16 +00:00
"title": Messages.kanban_newBoard,
"item": []
});
kanban.onChange();
});
2018-04-01 16:52:21 +00:00
2020-03-04 16:48:20 +00:00
var $container = $('#cp-app-kanban-content');
2020-03-09 13:27:33 +00:00
var $cContainer = $('#cp-app-kanban-container');
2020-03-04 16:48:20 +00:00
var addControls = function () {
2020-03-05 11:06:35 +00:00
// Quick or normal mode
2020-03-04 16:48:20 +00:00
var small = h('span.cp-kanban-view-small.fa.fa-minus');
var big = h('span.cp-kanban-view.fa.fa-bars');
$(small).click(function () {
2020-03-09 13:27:33 +00:00
if ($cContainer.hasClass('cp-kanban-quick')) { return; }
$cContainer.addClass('cp-kanban-quick');
//framework._.sfCommon.setPadAttribute('quickMode', true);
2020-03-04 16:48:20 +00:00
});
$(big).click(function () {
2020-03-09 13:27:33 +00:00
if (!$cContainer.hasClass('cp-kanban-quick')) { return; }
$cContainer.removeClass('cp-kanban-quick');
//framework._.sfCommon.setPadAttribute('quickMode', false);
2020-03-04 16:48:20 +00:00
});
2020-03-04 17:50:17 +00:00
2020-03-05 11:06:35 +00:00
// Tags filter
2020-03-04 17:50:17 +00:00
var existing = getExistingTags(kanban.options.boards);
2020-03-05 11:06:35 +00:00
var list = h('div.cp-kanban-filterTags-list');
2020-06-15 13:26:40 +00:00
var reset = h('button.btn.btn-cancel.cp-kanban-filterTags-reset', [
2020-06-15 13:16:33 +00:00
h('i.fa.fa-times'),
Messages.kanban_clearFilter
]);
var hint = h('span.cp-kanban-filterTags-name', Messages.kanban_tags);
2020-03-04 17:50:17 +00:00
var tags = h('div.cp-kanban-filterTags', [
h('span.cp-kanban-filterTags-toggle', [
hint,
reset,
]),
2020-03-12 14:43:10 +00:00
list,
2020-03-04 17:50:17 +00:00
]);
2020-03-05 11:06:35 +00:00
var $reset = $(reset);
var $list = $(list);
var $hint = $(hint);
var setTagFilterState = function (bool) {
2020-06-15 13:16:33 +00:00
$hint.css('visibility', bool? 'hidden': 'visible');
$reset.css('visibility', bool? 'visible': 'hidden');
};
setTagFilterState();
2020-03-05 11:06:35 +00:00
var getTags = function () {
return $list.find('span.active').map(function () {
return $(this).data('tag');
}).get();
};
2020-03-04 17:50:17 +00:00
var commitTags = function () {
2020-03-05 11:06:35 +00:00
var t = getTags();
setTagFilterState(t.length);
//framework._.sfCommon.setPadAttribute('tagsFilter', t);
2020-03-04 17:50:17 +00:00
kanban.options.tags = t;
kanban.setBoards(kanban.options.boards);
addEditItemButton(framework, kanban);
};
2020-03-05 11:06:35 +00:00
var redrawList = function (allTags) {
if (!Array.isArray(allTags)) { return; }
$list.empty();
2020-06-19 15:45:01 +00:00
$list.removeClass('cp-empty');
2020-03-05 13:06:34 +00:00
if (!allTags.length) {
2020-06-19 15:45:01 +00:00
$list.addClass('cp-empty');
2020-03-05 13:06:34 +00:00
$list.append(h('em', Messages.kanban_noTags));
return;
}
2020-03-05 11:06:35 +00:00
allTags.forEach(function (t) {
var tag;
$list.append(tag = h('span', {
'data-tag': t
}, t));
var $tag = $(tag).click(function () {
if ($tag.hasClass('active')) {
$tag.removeClass('active');
} else {
$tag.addClass('active');
}
commitTags();
});
});
};
redrawList(existing);
var setTags = function (tags) {
$list.find('span').removeClass('active');
if (!Array.isArray(tags)) { return; }
tags.forEach(function (t, i) {
if (existing.indexOf(t) === -1) {
// This tag doesn't exist anymore
tags.splice(i, 1);
return;
}
$list.find('span').filter(function () {
return $(this).data('tag') === t;
}).addClass('active');
});
setTagFilterState(tags.length);
//framework._.sfCommon.setPadAttribute('tagsFilter', tags);
2020-03-05 11:06:35 +00:00
};
setTagFilterState();
$reset.click(function () {
2020-03-05 11:06:35 +00:00
setTags([]);
commitTags();
});
2020-03-04 17:50:17 +00:00
2020-03-04 16:48:20 +00:00
var container = h('div#cp-kanban-controls', [
2020-03-05 11:06:35 +00:00
tags,
2020-03-04 16:48:20 +00:00
h('div.cp-kanban-changeView', [
small,
big
])
]);
2020-03-09 13:27:33 +00:00
$container.before(container);
2020-03-05 11:06:35 +00:00
2021-03-19 11:04:20 +00:00
var common = framework._.sfCommon;
var $button = common.createButton('toggle', true, {
element: $(container),
icon: 'fa-tags',
text: Messages.fm_tagsName,
2021-03-19 11:04:20 +00:00
}, function () {
$button.toggleClass('cp-toolbar-button-active');
2021-03-19 11:04:20 +00:00
});
$button.addClass('cp-toolbar-button-active');
2021-03-19 11:04:20 +00:00
framework._.toolbar.$bottomL.append($button);
2020-03-05 11:06:35 +00:00
onRedraw.reg(function () {
// Redraw if new tags have been added to items
var old = Sortify(existing);
var t = getTags();
existing = getExistingTags(kanban.options.boards);
if (old === Sortify(existing)) { return; } // No change
// New tags:
redrawList(existing);
setTags(t);
});
/*
2020-03-05 11:06:35 +00:00
framework._.sfCommon.getPadAttribute('tagsFilter', function (err, res) {
if (!err && Array.isArray(res)) {
setTags(res);
commitTags();
}
});
framework._.sfCommon.getPadAttribute('quickMode', function (err, res) {
if (!err && res) {
2020-03-09 13:27:33 +00:00
$cContainer.addClass('cp-kanban-quick');
2020-03-05 11:06:35 +00:00
}
});
*/
2020-03-04 16:48:20 +00:00
};
addControls();
return kanban;
};
2018-04-01 16:52:21 +00:00
2018-05-23 12:24:18 +00:00
var mkHelpMenu = function (framework) {
var $toolbarContainer = $('#cp-app-kanban-container');
2020-01-09 16:30:15 +00:00
2018-05-23 12:24:18 +00:00
var helpMenu = framework._.sfCommon.createHelpMenu(['kanban']);
$toolbarContainer.prepend(helpMenu.menu);
framework._.toolbar.$drawer.append(helpMenu.button);
};
// Start of the main loop
var andThen2 = function (framework) {
2018-04-01 16:52:21 +00:00
2018-05-22 16:22:25 +00:00
var kanban;
var $container = $('#cp-app-kanban-content');
2020-05-25 21:12:12 +00:00
var privateData = framework._.cpNfInner.metadataMgr.getPrivateData();
if (!privateData.isEmbed) {
mkHelpMenu(framework);
}
2018-05-23 12:24:18 +00:00
2021-02-08 10:06:32 +00:00
if (framework.isReadOnly() || framework.isLocked()) {
$container.addClass('cp-app-readonly');
}
2021-02-26 16:43:38 +00:00
framework.setFileImporter({accept: ['.json', 'application/json']}, function (content /*, file */) {
2021-02-26 09:38:06 +00:00
var parsed;
try { parsed = JSON.parse(content); }
catch (e) { return void console.error(e); }
2021-06-15 12:48:58 +00:00
if (parsed && parsed.id && parsed.lists && parsed.cards) {
return { content: Export.import(parsed) };
}
2021-02-26 09:38:06 +00:00
return { content: parsed };
});
2018-07-11 08:50:42 +00:00
framework.setFileExporter('.json', function () {
2018-10-18 16:50:38 +00:00
return new Blob([JSON.stringify(kanban.getBoardsJSON(), 0, 2)], {
2018-07-11 08:50:42 +00:00
type: 'application/json',
});
});
framework.onEditableChange(function (unlocked) {
if (framework.isReadOnly()) { return; }
2018-05-25 09:49:43 +00:00
if (!kanban) { return; }
if (unlocked) {
2020-03-02 11:05:09 +00:00
addEditItemButton(framework, kanban);
kanban.options.readOnly = false;
return void $container.removeClass('cp-app-readonly');
}
kanban.options.readOnly = true;
$container.addClass('cp-app-readonly');
$container.find('.kanban-edit-item').remove();
});
2018-04-01 16:52:21 +00:00
var getCursor = function () {
if (!kanban || !kanban.inEditMode) { return; }
try {
var id = kanban.inEditMode;
var newBoard;
var $el = $container.find('[data-id="'+id+'"]');
if (id === "new") {
$el = $container.find('.kanban-item.new-item');
newBoard = $el.closest('.kanban-board').attr('data-id');
} else if (!$el.length) {
$el = $container.find('[data-eid="'+id+'"]');
}
var isTop = $el && $el.hasClass('item-top');
if (!$el.length) { return; }
var $input = $el.find('input');
if (!$input.length) { return; }
var input = $input[0];
var val = ($input.val && $input.val()) || '';
var start = input.selectionStart;
var end = input.selectionEnd;
var json = kanban.getBoardJSON(id) || kanban.getItemJSON(id);
var oldVal = json && json.title;
return {
id: id,
newBoard: newBoard,
value: val,
start: start,
end: end,
isTop: isTop,
oldValue: oldVal
};
} catch (e) {
console.error(e);
return {};
}
};
var restoreCursor = function (data) {
2020-03-02 14:03:32 +00:00
if (!data) { return; }
try {
var id = data.id;
// An item was being added: add a new item
if (id === "new" && !data.oldValue) {
var $newBoard = $('.kanban-board[data-id="'+data.newBoard+'"]');
var topSelector = ':not([data-top])';
if (data.isTop) { topSelector = '[data-top]'; }
$newBoard.find('.kanban-title-button' + topSelector).click();
var $newInput = $newBoard.find('.kanban-item.new-item input');
$newInput.val(data.value);
$newInput[0].selectionStart = data.start;
$newInput[0].selectionEnd = data.end;
return;
}
// Edit a board title or a card title
var $el = $container.find('.kanban-board[data-id="'+id+'"]');
if (!$el.length) {
$el = $container.find('.kanban-item[data-eid="'+id+'"]');
}
if (!$el.length) { return; }
2020-03-03 17:35:56 +00:00
var isBoard = true;
var json = kanban.getBoardJSON(id);
if (!json) {
isBoard = false;
json = kanban.getItemJSON(id);
}
if (!json) { return; }
2020-03-03 17:35:56 +00:00
// Editing a board or card title...
$el.find(isBoard ? '.kanban-title-board' : '.kanban-item-text').click();
var $input = $el.find('input');
2020-03-03 17:35:56 +00:00
if (!$input.length) { return; }
// if the value was changed by a remote user, abort
setValueAndCursor($input[0], json.title, {
value: data.value,
selectionStart: data.start,
selectionEnd: data.end
});
} catch (e) {
console.error(e);
return;
}
};
framework.onContentUpdate(function (newContent) {
2018-05-22 16:22:25 +00:00
// Init if needed
if (!kanban) {
kanban = initKanban(framework, (newContent || {}).content);
2020-03-02 11:05:09 +00:00
addEditItemButton(framework, kanban);
2018-05-22 16:22:25 +00:00
return;
}
// Need to update the content
2018-05-17 08:27:16 +00:00
verbose("Content should be updated to " + newContent);
var currentContent = kanban.getBoardsJSON();
var remoteContent = newContent.content;
2018-05-16 10:22:43 +00:00
2018-05-22 16:22:25 +00:00
if (Sortify(currentContent) !== Sortify(remoteContent)) {
var cursor = getCursor();
verbose("Content is different.. Applying content");
kanban.options.boards = remoteContent;
updateBoards(framework, kanban, remoteContent);
restoreCursor(cursor);
2020-03-03 16:40:01 +00:00
onRemoteChange.fire();
}
2018-04-01 16:52:21 +00:00
});
framework.setContentGetter(function () {
2018-05-25 09:49:43 +00:00
if (!kanban) {
2020-03-04 17:50:17 +00:00
throw new Error("NOT INITIALIZED");
2018-05-25 09:49:43 +00:00
}
var content = kanban.getBoardsJSON();
2018-05-17 08:27:16 +00:00
verbose("Content current value is " + content);
return {
content: content
};
2018-04-01 16:52:21 +00:00
});
2020-03-02 11:05:09 +00:00
var cleanData = function (boards) {
if (typeof(boards) !== "object") { return; }
var items = boards.items || {};
var data = boards.data || {};
var list = boards.list || [];
// Remove duplicate boards
list = boards.list = Util.deduplicateString(list);
2020-03-02 11:05:09 +00:00
Object.keys(data).forEach(function (id) {
if (list.indexOf(Number(id)) === -1) {
list.push(Number(id));
}
// Remove duplicate items
var b = data[id];
b.item = Util.deduplicateString(b.item || []);
2020-03-02 11:05:09 +00:00
});
Object.keys(items).forEach(function (eid) {
var exists = Object.keys(data).some(function (id) {
2020-03-02 14:03:32 +00:00
return (data[id].item || []).indexOf(Number(eid)) !== -1;
2020-03-02 11:05:09 +00:00
});
if (!exists) { delete items[eid]; }
});
framework.localChange();
};
2018-05-16 10:22:43 +00:00
framework.onReady(function () {
$("#cp-app-kanban-content").focus();
2020-03-02 11:05:09 +00:00
var content = kanban.getBoardsJSON();
cleanData(content);
});
2018-04-01 16:52:21 +00:00
2018-05-25 09:49:43 +00:00
framework.onDefaultContentNeeded(function () {
kanban = initKanban(framework);
});
2020-03-09 17:49:57 +00:00
var myCursor = {};
onCursorUpdate.reg(function (data) {
myCursor = data;
framework.updateCursor();
});
framework.onCursorUpdate(function (data) {
if (!data) { return; }
if (data.reset) {
Object.keys(remoteCursors).forEach(function (id) {
if (remoteCursors[id].clear) {
remoteCursors[id].clear();
}
delete remoteCursors[id];
});
return;
}
2020-03-09 17:49:57 +00:00
var id = data.id;
// Clear existing cursor
Object.keys(remoteCursors).forEach(function (_id) {
if (_id.indexOf(id) === 0 && remoteCursors[_id].clear) {
remoteCursors[_id].clear();
delete remoteCursors[_id];
}
});
2020-03-09 17:49:57 +00:00
delete remoteCursors[id];
var cursor = data.cursor;
if (data.leave || !cursor) { return; }
if (!cursor.item && !cursor.board) { return; }
// Add new cursor
var avatar = getAvatar(cursor);
var $item = $('.kanban-item[data-eid="'+cursor.item+'"]');
if ($item.length) {
remoteCursors[id] = cursor;
$item.find('.cp-kanban-cursors').append(avatar);
return;
}
var $board = $('.kanban-board[data-id="'+cursor.board+'"]');
2020-03-09 17:49:57 +00:00
if ($board.length) {
remoteCursors[id] = cursor;
$board.find('header .cp-kanban-cursors').append(avatar);
}
});
framework.onCursorUpdate(function () {
if (!editModal || !editModal.conflict) { return; }
editModal.conflict.setValue();
});
framework.setCursorGetter(function () {
return myCursor;
});
framework.start();
2018-04-01 16:52:21 +00:00
};
var main = function () {
// var framework;
2018-04-01 16:52:21 +00:00
nThen(function (waitFor) {
// Framework initialization
Framework.create({
toolbarContainer: '#cme_toolbox',
contentContainer: '#cp-app-kanban-editor',
2018-05-16 10:22:43 +00:00
}, waitFor(function (framework) {
andThen2(framework);
2018-04-01 16:52:21 +00:00
}));
});
};
main();
});