Stackedit/public/res/extensions/documentManager.js

417 lines
17 KiB
JavaScript
Raw Normal View History

2013-08-11 00:52:27 +00:00
define([
"jquery",
"underscore",
2013-11-05 23:03:38 +00:00
"constants",
2013-08-11 00:52:27 +00:00
"utils",
2013-11-05 23:03:38 +00:00
"storage",
2013-08-11 00:52:27 +00:00
"classes/Extension",
"classes/FolderDescriptor",
2013-08-12 00:10:26 +00:00
"folderList",
2013-08-11 00:52:27 +00:00
"fileSystem",
2013-11-05 23:03:38 +00:00
], function($, _, constants, utils, storage, Extension, FolderDescriptor, folderList, fileSystem) {
2013-08-11 00:52:27 +00:00
var documentManager = new Extension("documentManager", 'Document Manager', false, true);
2013-11-07 23:10:38 +00:00
var fileMgr;
2013-08-11 00:52:27 +00:00
documentManager.onFileMgrCreated = function(fileMgrParameter) {
fileMgr = fileMgrParameter;
};
2013-11-07 23:10:38 +00:00
var eventMgr;
2013-08-11 00:52:27 +00:00
documentManager.onEventMgrCreated = function(eventMgrParameter) {
eventMgr = eventMgrParameter;
};
var folderEltTmpl = [
2013-08-12 22:08:22 +00:00
'<a href="#" class="list-group-item folder clearfix" data-folder-index="<%= folderDesc.folderIndex %>" data-toggle="collapse" data-target=".modal-document-manager .file-list.<%= id %>">',
2013-08-11 00:52:27 +00:00
'<label class="checkbox" title="Select"><input type="checkbox"></label>',
2013-08-12 00:10:26 +00:00
'<button class="btn btn-default button-delete" title="Delete"><i class="icon-trash"></i></button>',
2013-08-11 00:52:27 +00:00
'<button class="btn btn-default button-rename" title="Rename"><i class="icon-pencil"></i></button>',
2013-08-11 01:23:29 +00:00
'<div class="pull-right file-count"><%= _.size(folderDesc.fileList) %></div>',
2013-08-11 00:52:27 +00:00
'<div class="name"><i class="icon-folder"></i> ',
'<%= folderDesc.name %></div>',
'<input type="text" class="input-rename form-control hide"></a>',
2013-08-12 22:08:22 +00:00
'<div class="file-list collapse <%= id %> clearfix"><%= fileListHtml %></div>'
2013-08-11 00:52:27 +00:00
].join('');
2013-08-12 00:10:26 +00:00
var documentEltTmpl = [
2013-08-11 01:23:29 +00:00
'<li class="list-group-item file clearfix" data-file-index="<%= fileDesc.fileIndex %>">',
2013-08-11 00:52:27 +00:00
'<label class="checkbox" title="Select"><input type="checkbox"></label>',
2013-08-12 00:10:26 +00:00
'<button class="btn btn-default button-delete" title="Delete"><i class="icon-trash"></i></button>',
2013-08-11 00:52:27 +00:00
'<button class="btn btn-default button-rename" title="Rename"><i class="icon-pencil"></i></button>',
'<div class="name"><%= fileDesc.composeTitle() %></div>',
2013-08-11 01:23:29 +00:00
'<input type="text" class="input-rename form-control hide"></li>',
2013-08-11 00:52:27 +00:00
].join('');
2013-08-12 00:10:26 +00:00
var selectFolderEltTmpl = [
'<a href="#" class="list-group-item folder clearfix" data-folder-index="<%= folderDesc.folderIndex %>">',
'<div class="pull-right file-count"><%= _.size(folderDesc.fileList) %></div>',
'<div class="name"><i class="icon-forward"></i> ',
'<%= folderDesc.name %></div></a>',
].join('');
var selectedDocumentEltTmpl = [
'<li class="list-group-item file clearfix">',
'<div class="name"><%= fileDesc.composeTitle() %></div></li>',
].join('');
2013-11-07 23:10:38 +00:00
var isVisible;
var modalElt;
var documentListElt;
2013-08-12 00:10:26 +00:00
var selectedDocumentList = [];
var selectedFolderList = [];
function doSelect() {
selectedFolderList = [];
selectedDocumentList = [];
_.each(documentListElt.querySelectorAll('input[type="checkbox"]:checked'), function(checkboxElt) {
2013-08-22 00:19:59 +00:00
var $parentElt = $(checkboxElt.parentNode.parentNode);
var folderDesc = folderList[$parentElt.data('folderIndex')];
var fileDesc = fileSystem[$parentElt.data('fileIndex')];
2013-08-12 00:10:26 +00:00
if(folderDesc !== undefined) {
selectedFolderList.push(folderDesc);
}
else if(fileDesc !== undefined) {
selectedDocumentList.push(fileDesc);
}
});
}
2013-08-11 00:52:27 +00:00
2013-11-07 23:10:38 +00:00
var selectedDocumentListElt;
2013-08-12 00:10:26 +00:00
function doDeleteConfirmation() {
// Don't ask user confirmation if we delete only folders
if(_.size(selectedDocumentList) === 0) {
doDelete();
return;
}
2013-08-11 00:52:27 +00:00
2013-08-12 00:10:26 +00:00
// Build the selected document list
var selectedDocumentListHtml = _.chain(selectedDocumentList).sortBy(function(fileDesc) {
return fileDesc.title.toLowerCase();
}).reduce(function(result, fileDesc) {
return result + _.template(selectedDocumentEltTmpl, {
fileDesc: fileDesc,
});
}, '').value();
2013-08-15 23:17:16 +00:00
selectedDocumentListElt.innerHTML = '<ul class="file-list nav">' + selectedDocumentListHtml + '</ul>';
2013-08-12 00:10:26 +00:00
// Ask user confirmation
$(modalElt.querySelectorAll('.document-list')).addClass('hide');
$(modalElt.querySelectorAll('.confirm-delete, .selected-document-list')).removeClass('hide');
}
function doDelete() {
// Delete files
_.each(selectedDocumentList, function(fileDesc) {
fileDesc.folder && fileDesc.folder.removeFile(fileDesc);
fileMgr.deleteFile(fileDesc);
});
// Delete folders
_.each(selectedFolderList, function(folderDesc) {
utils.removeIndexFromArray("folder.list", folderDesc.folderIndex);
2013-11-05 23:03:38 +00:00
storage.removeItem(folderDesc.folderIndex + ".name");
storage.removeItem(folderDesc.folderIndex + ".files");
2013-08-12 00:10:26 +00:00
delete folderList[folderDesc.folderIndex];
});
eventMgr.onFoldersChanged();
}
2013-11-07 23:10:38 +00:00
var $liMoveElt;
var $liDeleteElt;
2013-08-12 00:10:26 +00:00
function doActiveButtons() {
doSelect();
2013-08-22 00:19:59 +00:00
$liMoveElt.toggleClass('disabled', _.size(folderList) === 0 || _.size(selectedDocumentList) === 0);
$liDeleteElt.toggleClass('disabled', _.size(selectedFolderList) === 0 && _.size(selectedDocumentList) === 0);
2013-08-12 00:10:26 +00:00
}
2013-11-07 23:10:38 +00:00
var orphanDocumentList;
var $documentCountElt;
var $folderCountElt;
2013-12-02 00:09:39 +00:00
var refreshManager = _.debounce(function() {
2013-08-12 00:10:26 +00:00
if(isVisible === false) {
return;
}
doActiveButtons();
// Refresh file/folder counters
2013-08-22 00:19:59 +00:00
$documentCountElt.text(_.size(fileSystem));
$folderCountElt.text(_.size(folderList) + 1);
2013-08-12 00:10:26 +00:00
// List orphan documents
orphanDocumentList = _.filter(fileSystem, function(fileDesc) {
2013-08-11 00:52:27 +00:00
return fileDesc.folder === undefined;
2013-08-12 00:10:26 +00:00
});
// Root folder
var documentListHtml = [
2013-08-12 22:08:22 +00:00
'<a href="#" class="list-group-item folder clearfix" data-toggle="collapse" data-target=".modal-document-manager .file-list.root-folder">',
2013-08-12 00:10:26 +00:00
'<label class="checkbox" title="Select"><input type="checkbox"></label>',
'<div class="pull-right file-count">',
_.size(orphanDocumentList),
'</div>',
'<div class="name"><i class="icon-folder"></i> ',
'ROOT folder</div></a>',
].join('');
// Add orphan documents
2013-08-12 22:08:22 +00:00
var orphanListHtml = _.chain(orphanDocumentList).sortBy(function(fileDesc) {
2013-08-11 00:52:27 +00:00
return fileDesc.title.toLowerCase();
}).reduce(function(result, fileDesc) {
2013-08-12 00:10:26 +00:00
return result + _.template(documentEltTmpl, {
2013-08-11 00:52:27 +00:00
fileDesc: fileDesc,
});
2013-08-12 22:08:22 +00:00
}, '').value();
orphanListHtml = orphanListHtml && '<ul class="nav">' + orphanListHtml + '</ul>';
documentListHtml += '<div class="file-list collapse root-folder clearfix">' + orphanListHtml + '</div>';
2013-08-11 00:52:27 +00:00
// Build directories
_.chain(folderList).sortBy(function(folderDesc) {
return folderDesc.name.toLowerCase();
}).each(function(folderDesc) {
var fileListHtml = _.chain(folderDesc.fileList).sortBy(function(fileDesc) {
return fileDesc.title.toLowerCase();
}).reduce(function(result, fileDesc) {
2013-08-12 00:10:26 +00:00
return result + _.template(documentEltTmpl, {
fileDesc: fileDesc,
});
2013-08-11 00:52:27 +00:00
}, '').value();
2013-08-12 22:08:22 +00:00
fileListHtml = fileListHtml && '<ul class="nav">' + fileListHtml + '</ul>';
2013-08-11 00:52:27 +00:00
documentListHtml += _.template(folderEltTmpl, {
folderDesc: folderDesc,
fileListHtml: fileListHtml,
id: folderDesc.folderIndex.replace('.', '')
});
});
documentListElt.innerHTML = documentListHtml;
2013-08-12 00:10:26 +00:00
// Set delete event listeners
_.each(documentListElt.querySelectorAll('.button-delete'), function(buttonElt) {
2013-08-22 00:19:59 +00:00
var $buttonElt = $(buttonElt);
$buttonElt.click(function(e) {
2013-08-12 00:10:26 +00:00
e.stopPropagation();
2013-08-22 00:19:59 +00:00
var $parentElt = $buttonElt.parent();
var folderDesc = folderList[$parentElt.data('folderIndex')];
var fileDesc = fileSystem[$parentElt.data('fileIndex')];
2013-08-12 00:10:26 +00:00
selectedDocumentList = [];
selectedFolderList = [];
if(folderDesc) {
selectedFolderList.push(folderDesc);
selectedDocumentList = folderDesc.fileList;
}
else if(fileDesc) {
selectedDocumentList.push(fileDesc);
}
doDeleteConfirmation();
});
});
2013-08-11 00:52:27 +00:00
// Set rename event listeners
_.each(documentListElt.querySelectorAll('.button-rename'), function(buttonElt) {
2013-08-22 00:19:59 +00:00
var $buttonElt = $(buttonElt);
$buttonElt.click(function(e) {
2013-08-12 00:10:26 +00:00
e.stopPropagation();
2013-08-22 00:19:59 +00:00
var $parentElt = $buttonElt.parent();
2013-11-07 23:10:38 +00:00
var name;
2013-08-22 00:19:59 +00:00
var folderDesc = folderList[$parentElt.data('folderIndex')];
var fileDesc = fileSystem[$parentElt.data('fileIndex')];
2013-08-11 00:52:27 +00:00
if(folderDesc) {
name = folderDesc.name;
}
else if(fileDesc) {
name = fileDesc.title;
}
2013-08-22 00:19:59 +00:00
$parentElt.find('.name').addClass('hide');
$parentElt.find('.input-rename').removeClass('hide').val(name)[0].select();
2013-08-11 00:52:27 +00:00
});
});
_.each(documentListElt.querySelectorAll('.input-rename'), function(inputElt) {
2013-08-22 00:19:59 +00:00
var $inputElt = $(inputElt);
2013-08-11 00:52:27 +00:00
function rename() {
2013-08-22 00:19:59 +00:00
var parentElt = $inputElt.parent();
var name = $.trim($inputElt.val());
2013-08-11 00:52:27 +00:00
var folderDesc = folderList[parentElt.data('folderIndex')];
var fileDesc = fileSystem[parentElt.data('fileIndex')];
if(name && folderDesc && name != folderDesc.name) {
folderDesc.name = name;
2013-08-13 23:30:50 +00:00
eventMgr.onFoldersChanged();
2013-08-11 00:52:27 +00:00
}
else if(name && fileDesc && name != fileDesc.title) {
fileDesc.title = name;
eventMgr.onTitleChanged(fileDesc);
}
else {
2013-08-22 00:19:59 +00:00
$inputElt.addClass('hide');
2013-08-11 00:52:27 +00:00
parentElt.find('.name').removeClass('hide');
}
}
2013-08-22 00:19:59 +00:00
$inputElt.blur(function() {
2013-08-11 00:52:27 +00:00
rename();
}).keyup(function(e) {
if(e.keyCode == 13) {
rename();
e.stopPropagation();
}
if(e.keyCode == 27) {
2013-08-22 00:19:59 +00:00
$inputElt.val('');
2013-08-11 00:52:27 +00:00
rename();
e.stopPropagation();
}
});
});
2013-10-06 14:34:40 +00:00
// Set file checkbox behavior
_.each(documentListElt.querySelectorAll('.file .checkbox'), function(checkboxElt) {
var $checkboxElt = $(checkboxElt);
$checkboxElt.click(function(e) {
e.stopPropagation();
}).find('[type=checkbox]').change(function() {
$checkboxElt.parents('.list-group').find('.folder [type=checkbox]').prop('checked', false);
});
});
// Set folder checkbox behavior
2013-08-12 00:10:26 +00:00
_.each(documentListElt.querySelectorAll('.folder .checkbox'), function(checkboxElt) {
2013-08-22 00:19:59 +00:00
var $checkboxElt = $(checkboxElt);
$checkboxElt.click(function(e) {
2013-08-12 00:10:26 +00:00
e.stopPropagation();
}).find('[type=checkbox]').change(function() {
2013-10-06 14:34:40 +00:00
$checkboxElt.parent().next().find('[type=checkbox]').prop('checked', this.checked);
2013-08-12 00:10:26 +00:00
});
2013-08-11 00:52:27 +00:00
});
2013-08-12 00:10:26 +00:00
// Set checkbox event listeners
$(documentListElt.querySelectorAll('[type=checkbox]')).change(doActiveButtons);
2013-12-02 00:09:39 +00:00
}, 50);
2013-08-11 00:52:27 +00:00
2013-08-12 00:10:26 +00:00
documentManager.onFileCreated = refreshManager;
documentManager.onFileDeleted = refreshManager;
documentManager.onTitleChanged = refreshManager;
documentManager.onSyncExportSuccess = refreshManager;
documentManager.onSyncRemoved = refreshManager;
documentManager.onNewPublishSuccess = refreshManager;
documentManager.onPublishRemoved = refreshManager;
2013-08-13 23:30:50 +00:00
documentManager.onFoldersChanged = refreshManager;
2013-08-12 00:10:26 +00:00
2013-08-11 00:52:27 +00:00
documentManager.onReady = function() {
2013-08-12 00:10:26 +00:00
modalElt = document.querySelector('.modal-document-manager');
2013-08-11 00:52:27 +00:00
documentListElt = modalElt.querySelector('.list-group.document-list');
2013-08-22 00:19:59 +00:00
$documentCountElt = $(modalElt.querySelectorAll('.document-count'));
$folderCountElt = $(modalElt.querySelectorAll('.folder-count'));
2013-08-12 00:10:26 +00:00
selectedDocumentListElt = modalElt.querySelector('.list-group.selected-document-list');
var selectFolderListElt = modalElt.querySelector('.list-group.select-folder-list');
// Only refresh manager if visible (costly)
$(modalElt).on('show.bs.modal', function() {
isVisible = true;
refreshManager();
}).on('hide.bs.modal', function() {
isVisible = false;
});
2013-08-11 00:52:27 +00:00
2013-08-12 00:10:26 +00:00
// Create folder action
$(modalElt.querySelectorAll('.action-create-folder')).click(function() {
2013-11-07 23:10:38 +00:00
var folderIndex;
2013-08-11 00:52:27 +00:00
do {
folderIndex = "folder." + utils.randomString();
} while (_.has(folderList, folderIndex));
2013-11-05 23:03:38 +00:00
storage[folderIndex + ".name"] = constants.DEFAULT_FOLDER_NAME;
2013-08-11 00:52:27 +00:00
// Create the folder descriptor
2013-11-05 23:03:38 +00:00
var folderDesc = new FolderDescriptor(folderIndex, constants.DEFAULT_FOLDER_NAME);
2013-08-11 00:52:27 +00:00
// Add the index to the folder list
utils.appendIndexToArray("folder.list", folderIndex);
folderList[folderIndex] = folderDesc;
2013-08-13 23:30:50 +00:00
eventMgr.onFoldersChanged();
2013-08-12 00:10:26 +00:00
// Edit the name when folder has just been created
2013-08-11 00:52:27 +00:00
var renameButtonElt = $(modalElt.querySelector('[data-folder-index="' + folderIndex + '"] .button-rename')).click();
2013-08-13 23:30:50 +00:00
modalElt.scrollTop += renameButtonElt.offset().top - 50;
2013-08-12 00:10:26 +00:00
});
// Selection dropdown menu actions
$(modalElt.querySelectorAll('.action-select-all')).click(function() {
2013-10-06 14:34:40 +00:00
$(documentListElt.querySelectorAll('input[type="checkbox"]')).prop('checked', true);
2013-08-12 00:10:26 +00:00
});
$(modalElt.querySelectorAll('.action-unselect-all')).click(function() {
2013-10-06 14:34:40 +00:00
$(documentListElt.querySelectorAll('input[type="checkbox"]')).prop('checked', false);
2013-08-11 00:52:27 +00:00
});
2013-08-12 00:10:26 +00:00
// Delete selection actions
2013-08-22 00:19:59 +00:00
var $aDeleteElt = $(modalElt.querySelectorAll('.action-delete-items')).click(function() {
if($liDeleteElt.hasClass('disabled')) {
2013-08-12 00:10:26 +00:00
return;
}
doSelect();
doDeleteConfirmation();
});
2013-08-22 00:19:59 +00:00
$liDeleteElt = $aDeleteElt.parent();
2013-08-12 00:10:26 +00:00
// Delete confirmation actions
$(modalElt.querySelectorAll('.action-delete-items-confirm')).click(function() {
doDelete();
$(modalElt.querySelectorAll('.document-list')).removeClass('hide');
$(modalElt.querySelectorAll('.confirm-delete, .selected-document-list')).addClass('hide');
});
// Move selection actions
2013-08-22 00:19:59 +00:00
var $aMoveElt = $(modalElt.querySelectorAll('.action-move-items')).click(function() {
if($liMoveElt.hasClass('disabled')) {
2013-08-12 00:10:26 +00:00
return;
}
doSelect();
// Build the destination folder list
var selectFolderListHtml = [
'<a href="#" class="list-group-item folder clearfix">',
'<div class="pull-right file-count">',
_.size(orphanDocumentList),
'</div>',
'<div class="name"><i class="icon-forward"></i> ',
'ROOT folder</div></a>',
].join('');
selectFolderListHtml += _.chain(folderList).sortBy(function(folderDesc) {
return folderDesc.name.toLowerCase();
}).reduce(function(result, folderDesc) {
return result + _.template(selectFolderEltTmpl, {
folderDesc: folderDesc,
});
}, '').value();
selectFolderListElt.innerHTML = selectFolderListHtml;
// Set selection event listeners
_.each(selectFolderListElt.querySelectorAll('.folder'), function(folderElt) {
folderElt = $(folderElt);
folderElt.click(function() {
var folderDesc = folderList[folderElt.data('folderIndex')];
_.each(selectedDocumentList, function(fileDesc) {
fileDesc.folder && fileDesc.folder.removeFile(fileDesc);
folderDesc && folderDesc.addFile(fileDesc);
});
eventMgr.onFoldersChanged();
$(modalElt.querySelectorAll('.document-list')).removeClass('hide');
$(modalElt.querySelectorAll('.choose-folder, .select-folder-list')).addClass('hide');
});
});
// Ask user for destination folder
$(modalElt.querySelectorAll('.document-list')).addClass('hide');
$(modalElt.querySelectorAll('.choose-folder, .select-folder-list')).removeClass('hide');
});
2013-08-22 00:19:59 +00:00
$liMoveElt = $aMoveElt.parent();
2013-08-12 00:10:26 +00:00
// Cancel button
$(modalElt.querySelectorAll('.action-cancel')).click(function() {
$(modalElt.querySelectorAll('.document-list')).removeClass('hide');
$(modalElt.querySelectorAll('.confirm-delete, .choose-folder, .selected-document-list, .select-folder-list')).addClass('hide');
});
2013-08-11 00:52:27 +00:00
};
return documentManager;
});