2020-10-14 21:43:52 +00:00
|
|
|
import _ from 'lodash';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* NOTE: Store should contain all methods that work with items. At the moment
|
|
|
|
* there are still a few places where we change the items from the outside,
|
|
|
|
* while it's very important that the items behave a certain way.
|
|
|
|
*/
|
|
|
|
|
|
|
|
function Store(inputData) {
|
|
|
|
let idList = [];
|
|
|
|
let values = {};
|
2020-10-15 20:38:04 +00:00
|
|
|
let changed = false;
|
2020-10-14 21:43:52 +00:00
|
|
|
|
|
|
|
let ID = function () {
|
|
|
|
return '_' + Math.random().toString(36).substr(2, 12);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {int} index
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function currentID(index) {
|
|
|
|
if (index === idList.length) {
|
|
|
|
return 'at-end'
|
|
|
|
}
|
|
|
|
return idList[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} id
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
|
|
|
function index(id) {
|
|
|
|
return _.findIndex(idList, value => value === id)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} id
|
|
|
|
* @return {object}
|
|
|
|
*/
|
|
|
|
function value(id) {
|
|
|
|
return values[id];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} afterId
|
|
|
|
* @return {object}
|
|
|
|
*/
|
|
|
|
function afterValue(afterId) {
|
|
|
|
let i = index(afterId)
|
|
|
|
return values[idList[i + 1]]
|
|
|
|
}
|
|
|
|
|
|
|
|
function prevCursorPosition(cursor) {
|
|
|
|
let curIndent = values[idList[cursor]].indented
|
|
|
|
let curClosed = values[idList[cursor]].fold !== 'open';
|
|
|
|
if (!curClosed) {
|
|
|
|
curIndent = 10000000;
|
|
|
|
}
|
|
|
|
let moving = true
|
|
|
|
|
|
|
|
while (moving) {
|
|
|
|
cursor--
|
|
|
|
if (cursor < 0) {
|
|
|
|
cursor = idList.length - 1
|
|
|
|
curIndent = values[idList[cursor]].indented
|
|
|
|
}
|
|
|
|
let next = values[idList[cursor]];
|
|
|
|
if (curIndent >= next.indented && !next.hidden) {
|
|
|
|
moving = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return cursor
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find the next 'open' position in the list.
|
|
|
|
*
|
|
|
|
* @param {number} cursor
|
|
|
|
* @param {bool} wrap
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
|
|
|
function nextCursorPosition(cursor, wrap) {
|
|
|
|
let curIndent = values[idList[cursor]].indented
|
|
|
|
let curClosed = values[idList[cursor]].fold !== 'open';
|
|
|
|
if (!curClosed) {
|
|
|
|
curIndent = 10000000;
|
|
|
|
}
|
|
|
|
let moving = true
|
|
|
|
|
|
|
|
while (moving) {
|
|
|
|
cursor++
|
|
|
|
if (wrap) {
|
|
|
|
if (cursor >= idList.length) {
|
|
|
|
cursor = 0
|
|
|
|
curIndent = 0
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return cursor
|
|
|
|
}
|
|
|
|
let next = values[idList[cursor]];
|
|
|
|
if (curIndent >= next.indented) {
|
|
|
|
moving = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return cursor
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} beforeId
|
|
|
|
* @param {object} item
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function insertBefore(beforeId, item) {
|
|
|
|
let index = _.findIndex(idList, (id) => id === beforeId)
|
|
|
|
let id = item.id
|
|
|
|
if (!id) {
|
|
|
|
let newId = ID()
|
|
|
|
item.id = newId
|
|
|
|
values[newId] = item
|
|
|
|
id = newId
|
|
|
|
}
|
2020-10-15 20:47:51 +00:00
|
|
|
if (!beforeId) {
|
|
|
|
idList.push(id)
|
|
|
|
} else {
|
|
|
|
idList.splice(index, 0, id)
|
|
|
|
}
|
2020-10-15 20:38:04 +00:00
|
|
|
changed = true
|
2020-10-14 21:43:52 +00:00
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} afterId
|
|
|
|
* @param {object} items
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function insertAfter(afterId, ...items) {
|
|
|
|
if (afterId === 'at-end') {
|
|
|
|
let newItems = _.map(items, item => {
|
|
|
|
return this.append(item)
|
|
|
|
})
|
|
|
|
return newItems[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
let index = _.findIndex(idList, (id) => id === afterId)
|
|
|
|
|
|
|
|
let newItems = _.map(items, item => {
|
|
|
|
let id = item.id;
|
|
|
|
if (!id) {
|
|
|
|
let newId = ID()
|
|
|
|
item.id = newId
|
|
|
|
values[newId] = item
|
|
|
|
}
|
|
|
|
return item.id
|
|
|
|
})
|
|
|
|
idList.splice(index + 1, 0, ...newItems)
|
2020-10-15 20:38:04 +00:00
|
|
|
changed = true
|
2020-10-14 21:43:52 +00:00
|
|
|
return newItems[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {object} item
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function append(item) {
|
|
|
|
let id = item.id;
|
|
|
|
if (!item.id || item.id[0] !== '_') {
|
|
|
|
let newId = ID();
|
|
|
|
item.id = newId
|
|
|
|
id = newId
|
|
|
|
}
|
|
|
|
delete item.children
|
|
|
|
values[id] = item;
|
|
|
|
idList.push(id)
|
2020-10-15 20:38:04 +00:00
|
|
|
changed = true
|
2020-10-14 21:43:52 +00:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @callback updateCallback
|
|
|
|
* @param {object} item
|
|
|
|
* @param {object} prev
|
|
|
|
* @param {object} next
|
|
|
|
* @return {object}
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @param {string} currentId
|
|
|
|
* @param {updateCallback} callback
|
|
|
|
* @returns {*}
|
|
|
|
*/
|
|
|
|
function update(currentId, callback) {
|
|
|
|
let index = _.findIndex(idList, (id) => id === currentId)
|
2020-10-15 20:38:04 +00:00
|
|
|
let oldValue = _.clone(values[currentId])
|
|
|
|
let newValue = callback(values[currentId], values[idList[index - 1]], values[idList[index + 1]]);
|
2020-10-21 18:49:23 +00:00
|
|
|
if (oldValue && oldValue.text !== newValue.text) {
|
2020-10-15 20:38:04 +00:00
|
|
|
values[currentId] = newValue
|
|
|
|
changed = true
|
|
|
|
}
|
2020-10-14 21:43:52 +00:00
|
|
|
return currentId
|
|
|
|
}
|
|
|
|
|
|
|
|
function length() {
|
|
|
|
return idList.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns {{result: Array, values: {}, idList: []}}
|
|
|
|
*/
|
|
|
|
function debug() {
|
|
|
|
let result = _.map(idList, (id) => {
|
|
|
|
return values[id];
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
result,
|
|
|
|
idList,
|
|
|
|
values
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function remove(start, len) {
|
|
|
|
idList.splice(start, len)
|
2020-10-15 20:38:04 +00:00
|
|
|
changed = true
|
2020-10-14 21:43:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {int} start
|
|
|
|
* @param {int} end
|
|
|
|
* @returns {string[]}
|
|
|
|
*/
|
|
|
|
function slice(start, end) {
|
|
|
|
return idList.slice(start, end)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {int} first
|
|
|
|
* @param {int} len
|
|
|
|
* @param {int} dir
|
|
|
|
*/
|
|
|
|
function indent(first, len, dir) {
|
|
|
|
let selection = idList.slice(first, first + len)
|
|
|
|
|
|
|
|
let result = _.reduce(selection, function (list, itemId) {
|
|
|
|
let item = values[itemId]
|
|
|
|
if (first === 0 && list.length === 0) {
|
|
|
|
values[itemId].indented = 0
|
|
|
|
} else {
|
|
|
|
let indent = item.indented + dir
|
|
|
|
let prevIndent =
|
|
|
|
list.length === 0
|
|
|
|
? values[idList[first - 1]].indented
|
|
|
|
: _.last(list).indented
|
|
|
|
|
|
|
|
indent = Math.max(0, indent)
|
|
|
|
|
|
|
|
if (list.length === 0) {
|
|
|
|
values[idList[first - 1]].fold = 'open'
|
|
|
|
} else {
|
|
|
|
_.last(list).fold = 'open'
|
|
|
|
}
|
|
|
|
|
|
|
|
if (indent < prevIndent || Math.abs(prevIndent - indent) <= 1) {
|
|
|
|
values[itemId].indented = indent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
list.push(item)
|
|
|
|
return list
|
|
|
|
}, [])
|
2020-10-15 20:38:04 +00:00
|
|
|
changed = true
|
2020-10-14 21:43:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} from
|
|
|
|
* @param {string} to
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
function moveBefore(from, to) {
|
|
|
|
let fromIndex = _.findIndex(idList, (id) => id === from)
|
|
|
|
let item = values[from]
|
|
|
|
remove(fromIndex, 1)
|
2020-10-15 20:38:04 +00:00
|
|
|
let result = insertBefore(to, item)
|
|
|
|
changed = true
|
|
|
|
return result
|
2020-10-14 21:43:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Number} first
|
|
|
|
* @param {Number} indent
|
|
|
|
* @returns {Number}
|
|
|
|
*/
|
|
|
|
function firstSameIndented(first, indent) {
|
|
|
|
if (first <= 0 || first >= idList.length) return first
|
|
|
|
first--
|
|
|
|
while (first > 0 && values[idList[first]].indented > indent) {
|
|
|
|
first--
|
|
|
|
}
|
|
|
|
return first
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {int} first
|
|
|
|
* @returns {int}
|
|
|
|
*/
|
|
|
|
function lastHigherIndented(first) {
|
|
|
|
if (first < 0 || first >= idList.length) return first
|
|
|
|
let minLevel = values[idList[first]].indented
|
|
|
|
first++;
|
|
|
|
while (first !== idList.length && values[idList[first]].indented > minLevel) {
|
|
|
|
first++
|
|
|
|
}
|
|
|
|
return first
|
|
|
|
}
|
|
|
|
|
|
|
|
function selectItemsFrom(from) {
|
|
|
|
if (!from) {
|
|
|
|
return _.map(idList, id => values[id])
|
|
|
|
}
|
|
|
|
|
|
|
|
let first = _.findIndex(idList, id => id === from)
|
|
|
|
let items = _.map(idList.slice(first + 1), id => values[id])
|
|
|
|
return [values[from], ..._.takeWhile(items, item => item.indented > values[from].indented)]
|
|
|
|
}
|
|
|
|
|
|
|
|
function flat(from) {
|
|
|
|
return selectItemsFrom(from)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a tree starting at from node down.
|
|
|
|
* @param from
|
|
|
|
* @returns {*}
|
|
|
|
*/
|
|
|
|
function tree(from) {
|
|
|
|
let items = selectItemsFrom(from)
|
|
|
|
|
|
|
|
let top = (stack) => {
|
|
|
|
return stack[stack.length - 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
let pop = (stack) => {
|
|
|
|
stack.pop()
|
|
|
|
}
|
|
|
|
let push = (stack, item) => {
|
|
|
|
stack.push(item)
|
|
|
|
}
|
|
|
|
|
|
|
|
const stack = _.reduce(items, (stack, item) => {
|
|
|
|
if (stack.length === 0) {
|
2020-10-25 14:18:39 +00:00
|
|
|
// console.log(`push ${item.text} on stack`)
|
2020-10-14 21:43:52 +00:00
|
|
|
return [[item]]
|
|
|
|
}
|
|
|
|
|
2020-10-25 14:18:39 +00:00
|
|
|
let stackIndented = top(stack)[0].indented
|
2020-10-14 21:43:52 +00:00
|
|
|
const itemIndented = item.indented
|
|
|
|
|
|
|
|
if (itemIndented > stackIndented) {
|
2020-10-25 14:18:39 +00:00
|
|
|
push(stack, [Object.assign({}, item)]) // [ ... ] => [ ..., item ]
|
|
|
|
// console.log(`push ${item.text} on stack`)
|
2020-10-14 21:43:52 +00:00
|
|
|
} else {
|
2020-10-25 14:18:39 +00:00
|
|
|
// console.log(`itemIndented = ${itemIndented}, stackIndented = ${stackIndented}`)
|
2020-10-14 21:43:52 +00:00
|
|
|
while (stack.length > 1 && itemIndented < stackIndented) {
|
|
|
|
let children = top(stack)
|
2020-10-25 14:18:39 +00:00
|
|
|
pop(stack) // [ ..., A ] => [ ... ]
|
|
|
|
stackIndented = top(stack)[0].indented
|
|
|
|
// console.log(`pop stack`)
|
2020-10-14 21:43:52 +00:00
|
|
|
let cur = top(stack)
|
|
|
|
cur[cur.length - 1].children = children
|
2020-10-25 14:18:39 +00:00
|
|
|
// console.log(`set children for ${cur[cur.length-1].text}`)
|
|
|
|
}
|
|
|
|
if (itemIndented === stackIndented) {
|
|
|
|
top(stack).push(Object.assign({}, item))
|
|
|
|
// console.log(`add child ${item.text}`)
|
|
|
|
} else {
|
|
|
|
push(stack, [Object.assign({}, item)])
|
|
|
|
// console.log(`push ${item.text} on stack`)
|
2020-10-14 21:43:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return stack
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
while (stack.length > 1) {
|
|
|
|
let children = top(stack)
|
|
|
|
pop(stack)
|
|
|
|
let item = top(stack)
|
|
|
|
item[item.length - 1].children = children
|
|
|
|
}
|
|
|
|
|
|
|
|
return top(stack)
|
|
|
|
}
|
|
|
|
|
2020-10-15 20:38:04 +00:00
|
|
|
function clearChanged() {
|
|
|
|
changed = false
|
|
|
|
}
|
|
|
|
|
|
|
|
function hasChanged() {
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
2020-10-28 19:01:25 +00:00
|
|
|
function replaceChildren(id, newChildren) {
|
|
|
|
let first = index(id)
|
|
|
|
let firstVal = value(id)
|
|
|
|
if (firstVal.fleeting) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
let last = lastHigherIndented(first)
|
|
|
|
first++
|
|
|
|
let ids = []
|
|
|
|
_.each(newChildren, item => {
|
|
|
|
item.indented += 1 + firstVal.indented
|
|
|
|
if (!item.id) {
|
|
|
|
item.id = ID()
|
|
|
|
}
|
|
|
|
ids.push(item.id)
|
|
|
|
values[item.id] = item
|
|
|
|
})
|
|
|
|
idList.splice(first, last - first, ...ids)
|
|
|
|
}
|
|
|
|
|
|
|
|
let removeLevel = 9999999999;
|
2020-10-14 21:43:52 +00:00
|
|
|
_.each(inputData, (d) => {
|
2020-10-28 19:01:25 +00:00
|
|
|
if (d.text.startsWith("{{query:")) {
|
|
|
|
removeLevel = d.indented;
|
|
|
|
append(d)
|
|
|
|
} else if (d.indented <= removeLevel) {
|
|
|
|
removeLevel = 9999999999;
|
|
|
|
append(d)
|
|
|
|
}
|
2020-10-14 21:43:52 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return {
|
|
|
|
currentID,
|
|
|
|
index,
|
|
|
|
value,
|
|
|
|
afterValue,
|
|
|
|
insertBefore,
|
|
|
|
insertAfter,
|
|
|
|
indent,
|
|
|
|
append,
|
|
|
|
update,
|
|
|
|
length,
|
|
|
|
slice,
|
|
|
|
remove,
|
|
|
|
moveBefore,
|
|
|
|
firstSameIndented,
|
|
|
|
lastHigherIndented,
|
|
|
|
prevCursorPosition,
|
|
|
|
nextCursorPosition,
|
|
|
|
debug,
|
|
|
|
tree,
|
2020-10-15 20:38:04 +00:00
|
|
|
flat,
|
2020-10-28 19:01:25 +00:00
|
|
|
replaceChildren,
|
2020-10-15 20:38:04 +00:00
|
|
|
hasChanged,
|
|
|
|
clearChanged,
|
2020-10-14 21:43:52 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Store;
|