2021-01-22 17:41:11 +00:00
|
|
|
const Folder = require('../../models/Folder').default;
|
2021-08-16 16:18:32 +01:00
|
|
|
const Setting = require('../../models/Setting').default;
|
2020-11-05 16:58:23 +00:00
|
|
|
const BaseModel = require('../../BaseModel').default;
|
2018-05-06 12:11:59 +01:00
|
|
|
|
2020-03-13 23:46:14 +00:00
|
|
|
const shared = {};
|
2017-11-05 23:55:01 +00:00
|
|
|
|
2018-05-06 12:11:59 +01:00
|
|
|
function folderHasChildren_(folders, folderId) {
|
|
|
|
for (let i = 0; i < folders.length; i++) {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder = folders[i];
|
2018-05-06 12:11:59 +01:00
|
|
|
if (folder.parent_id === folderId) return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-05-09 10:49:31 +01:00
|
|
|
function folderIsVisible(folders, folderId, collapsedFolderIds) {
|
|
|
|
if (!collapsedFolderIds || !collapsedFolderIds.length) return true;
|
|
|
|
|
|
|
|
while (true) {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder = BaseModel.byId(folders, folderId);
|
2019-09-19 22:51:18 +01:00
|
|
|
if (!folder) throw new Error(`No folder with id ${folder.id}`);
|
2018-05-09 10:49:31 +01:00
|
|
|
if (!folder.parent_id) return true;
|
|
|
|
if (collapsedFolderIds.indexOf(folder.parent_id) >= 0) return false;
|
|
|
|
folderId = folder.parent_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-26 18:04:32 +00:00
|
|
|
function renderFoldersRecursive_(props, renderItem, items, parentId, depth, order) {
|
2018-05-06 12:11:59 +01:00
|
|
|
const folders = props.folders;
|
|
|
|
for (let i = 0; i < folders.length; i++) {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder = folders[i];
|
2018-05-09 09:53:47 +01:00
|
|
|
if (!Folder.idsEqual(folder.parent_id, parentId)) continue;
|
2018-05-09 10:49:31 +01:00
|
|
|
if (!folderIsVisible(props.folders, folder.id, props.collapsedFolderIds)) continue;
|
2018-05-06 12:11:59 +01:00
|
|
|
const hasChildren = folderHasChildren_(folders, folder.id);
|
2019-01-26 18:04:32 +00:00
|
|
|
order.push(folder.id);
|
2018-05-06 12:11:59 +01:00
|
|
|
items.push(renderItem(folder, props.selectedFolderId == folder.id && props.notesParentType == 'Folder', hasChildren, depth));
|
2019-01-26 18:04:32 +00:00
|
|
|
if (hasChildren) {
|
|
|
|
const result = renderFoldersRecursive_(props, renderItem, items, folder.id, depth + 1, order);
|
|
|
|
items = result.items;
|
|
|
|
order = result.order;
|
|
|
|
}
|
2017-11-05 23:55:01 +00:00
|
|
|
}
|
2019-01-26 18:04:32 +00:00
|
|
|
return {
|
|
|
|
items: items,
|
|
|
|
order: order,
|
|
|
|
};
|
2017-11-05 23:55:01 +00:00
|
|
|
}
|
|
|
|
|
2018-05-06 12:11:59 +01:00
|
|
|
shared.renderFolders = function(props, renderItem) {
|
2019-01-26 18:04:32 +00:00
|
|
|
return renderFoldersRecursive_(props, renderItem, [], '', 0, []);
|
2019-07-29 15:43:53 +02:00
|
|
|
};
|
2018-05-06 12:11:59 +01:00
|
|
|
|
2017-11-05 23:55:01 +00:00
|
|
|
shared.renderTags = function(props, renderItem) {
|
2020-03-13 23:46:14 +00:00
|
|
|
const tags = props.tags.slice();
|
2019-07-29 15:43:53 +02:00
|
|
|
tags.sort((a, b) => {
|
2020-11-06 10:50:44 +00:00
|
|
|
// It seems title can sometimes be undefined (perhaps when syncing
|
|
|
|
// and before tag has been decrypted?). It would be best to find
|
|
|
|
// the root cause but for now that will do.
|
|
|
|
//
|
|
|
|
// Fixes https://github.com/laurent22/joplin/issues/4051
|
|
|
|
if (!a || !a.title || !b || !b.title) return 0;
|
|
|
|
|
2020-10-23 15:48:11 +01:00
|
|
|
// Note: while newly created tags are normalized and lowercase
|
|
|
|
// imported tags might be any case, so we need to do case-insensitive
|
|
|
|
// sort.
|
|
|
|
return a.title.toLowerCase() < b.title.toLowerCase() ? -1 : +1;
|
2019-07-29 15:43:53 +02:00
|
|
|
});
|
2020-03-13 23:46:14 +00:00
|
|
|
const tagItems = [];
|
2019-01-26 18:04:32 +00:00
|
|
|
const order = [];
|
2017-11-05 23:55:01 +00:00
|
|
|
for (let i = 0; i < tags.length; i++) {
|
|
|
|
const tag = tags[i];
|
2019-01-26 18:04:32 +00:00
|
|
|
order.push(tag.id);
|
2017-11-05 23:55:01 +00:00
|
|
|
tagItems.push(renderItem(tag, props.selectedTagId == tag.id && props.notesParentType == 'Tag'));
|
|
|
|
}
|
2019-01-26 18:04:32 +00:00
|
|
|
return {
|
|
|
|
items: tagItems,
|
|
|
|
order: order,
|
|
|
|
};
|
2019-07-29 15:43:53 +02:00
|
|
|
};
|
2017-11-05 23:55:01 +00:00
|
|
|
|
2017-11-06 21:11:15 +00:00
|
|
|
shared.synchronize_press = async function(comp) {
|
2020-11-05 16:58:23 +00:00
|
|
|
const { reg } = require('../../registry.js');
|
2017-11-06 21:11:15 +00:00
|
|
|
|
|
|
|
const action = comp.props.syncStarted ? 'cancel' : 'start';
|
|
|
|
|
2021-08-16 16:18:32 +01:00
|
|
|
if (!Setting.value('sync.target')) {
|
|
|
|
comp.props.dispatch({
|
|
|
|
type: 'SIDE_MENU_CLOSE',
|
|
|
|
});
|
|
|
|
|
|
|
|
comp.props.dispatch({
|
|
|
|
type: 'NAV_GO',
|
|
|
|
routeName: 'Config',
|
|
|
|
sectionName: 'sync',
|
|
|
|
});
|
|
|
|
|
|
|
|
return 'init';
|
|
|
|
}
|
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
if (!(await reg.syncTarget().isAuthenticated())) {
|
2018-01-25 19:01:14 +00:00
|
|
|
if (reg.syncTarget().authRouteName()) {
|
|
|
|
comp.props.dispatch({
|
|
|
|
type: 'NAV_GO',
|
|
|
|
routeName: reg.syncTarget().authRouteName(),
|
|
|
|
});
|
|
|
|
return 'auth';
|
|
|
|
}
|
|
|
|
|
|
|
|
reg.logger().info('Not authentified with sync target - please check your credential.');
|
|
|
|
return 'error';
|
2017-11-06 21:11:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let sync = null;
|
|
|
|
try {
|
2017-11-24 18:06:30 +00:00
|
|
|
sync = await reg.syncTarget().synchronizer();
|
2017-11-06 21:11:15 +00:00
|
|
|
} catch (error) {
|
|
|
|
reg.logger().info('Could not acquire synchroniser:');
|
|
|
|
reg.logger().info(error);
|
|
|
|
return 'error';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (action == 'cancel') {
|
|
|
|
sync.cancel();
|
|
|
|
return 'cancel';
|
|
|
|
} else {
|
|
|
|
reg.scheduleSync(0);
|
|
|
|
return 'sync';
|
|
|
|
}
|
2019-07-29 15:43:53 +02:00
|
|
|
};
|
2017-11-06 21:11:15 +00:00
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
module.exports = shared;
|