2609 lines
94 KiB
JavaScript
2609 lines
94 KiB
JavaScript
/**
|
|
* ownCloud
|
|
*
|
|
* @author Vincent Petry
|
|
* @copyright 2014 Vincent Petry <pvince81@owncloud.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
|
|
* License as published by the Free Software Foundation; either
|
|
* version 3 of the License, or any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU AFFERO GENERAL PUBLIC LICENSE for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
describe('OCA.Files.FileList tests', function() {
|
|
var FileInfo = OC.Files.FileInfo;
|
|
var testFiles, testRoot, notificationStub, fileList, pageSizeStub;
|
|
var bcResizeStub;
|
|
var filesClient;
|
|
var redirectStub;
|
|
|
|
/**
|
|
* Generate test file data
|
|
*/
|
|
function generateFiles(startIndex, endIndex) {
|
|
var files = [];
|
|
var name;
|
|
for (var i = startIndex; i <= endIndex; i++) {
|
|
name = 'File with index ';
|
|
if (i < 10) {
|
|
// do not rely on localeCompare here
|
|
// and make the sorting predictable
|
|
// cross-browser
|
|
name += '0';
|
|
}
|
|
name += i + '.txt';
|
|
files.push(new FileInfo({
|
|
id: i,
|
|
type: 'file',
|
|
name: name,
|
|
mimetype: 'text/plain',
|
|
size: i * 2,
|
|
etag: 'abc'
|
|
}));
|
|
}
|
|
return files;
|
|
}
|
|
|
|
beforeEach(function() {
|
|
filesClient = new OC.Files.Client({
|
|
host: 'localhost',
|
|
port: 80,
|
|
// FIXME: uncomment after fixing the test OC.webroot
|
|
//root: OC.webroot + '/remote.php/webdav',
|
|
root: '/remote.php/webdav',
|
|
useHTTPS: false
|
|
});
|
|
redirectStub = sinon.stub(OC, 'redirect');
|
|
notificationStub = sinon.stub(OC.Notification, 'showTemporary');
|
|
// prevent resize algo to mess up breadcrumb order while
|
|
// testing
|
|
bcResizeStub = sinon.stub(OCA.Files.BreadCrumb.prototype, '_resize');
|
|
|
|
// init parameters and test table elements
|
|
$('#testArea').append(
|
|
'<div id="app-content-files">' +
|
|
// init horrible parameters
|
|
'<input type="hidden" id="dir" value="/subdir"/>' +
|
|
'<input type="hidden" id="permissions" value="31"/>' +
|
|
// dummy controls
|
|
'<div id="controls">' +
|
|
' <div class="actions creatable"></div>' +
|
|
' <div class="notCreatable"></div>' +
|
|
'</div>' +
|
|
// uploader
|
|
'<input type="file" id="file_upload_start" name="files[]" multiple="multiple">' +
|
|
// dummy table
|
|
// TODO: at some point this will be rendered by the fileList class itself!
|
|
'<table id="filestable">' +
|
|
'<thead><tr>' +
|
|
'<th id="headerName" class="hidden column-name">' +
|
|
'<input type="checkbox" id="select_all_files" class="select-all checkbox">' +
|
|
'<a class="name columntitle" data-sort="name"><span>Name</span><span class="sort-indicator"></span></a>' +
|
|
'<span id="selectedActionsList" class="selectedActions hidden">' +
|
|
'<a href class="download"><img src="actions/download.svg">Download</a>' +
|
|
'<a href class="delete-selected">Delete</a></span>' +
|
|
'</th>' +
|
|
'<th class="hidden column-size"><a class="columntitle" data-sort="size"><span class="sort-indicator"></span></a></th>' +
|
|
'<th class="hidden column-mtime"><a class="columntitle" data-sort="mtime"><span class="sort-indicator"></span></a></th>' +
|
|
'</tr></thead>' +
|
|
'<tbody id="fileList"></tbody>' +
|
|
'<tfoot></tfoot>' +
|
|
'</table>' +
|
|
// TODO: move to handlebars template
|
|
'<div id="emptycontent"><h2>Empty content message</h2><p class="uploadmessage">Upload message</p></div>' +
|
|
'<div class="nofilterresults hidden"></div>' +
|
|
'</div>'
|
|
);
|
|
|
|
testRoot = new FileInfo({
|
|
// root entry
|
|
id: 99,
|
|
type: 'dir',
|
|
name: '/subdir',
|
|
mimetype: 'httpd/unix-directory',
|
|
size: 1200000,
|
|
etag: 'a0b0c0d0',
|
|
permissions: OC.PERMISSION_ALL
|
|
});
|
|
testFiles = [new FileInfo({
|
|
id: 1,
|
|
type: 'file',
|
|
name: 'One.txt',
|
|
mimetype: 'text/plain',
|
|
mtime: 123456789,
|
|
size: 12,
|
|
etag: 'abc',
|
|
permissions: OC.PERMISSION_ALL
|
|
}), new FileInfo({
|
|
id: 2,
|
|
type: 'file',
|
|
name: 'Two.jpg',
|
|
mimetype: 'image/jpeg',
|
|
mtime: 234567890,
|
|
size: 12049,
|
|
etag: 'def',
|
|
permissions: OC.PERMISSION_ALL
|
|
}), new FileInfo({
|
|
id: 3,
|
|
type: 'file',
|
|
name: 'Three.pdf',
|
|
mimetype: 'application/pdf',
|
|
mtime: 234560000,
|
|
size: 58009,
|
|
etag: '123',
|
|
permissions: OC.PERMISSION_ALL
|
|
}), new FileInfo({
|
|
id: 4,
|
|
type: 'dir',
|
|
name: 'somedir',
|
|
mimetype: 'httpd/unix-directory',
|
|
mtime: 134560000,
|
|
size: 250,
|
|
etag: '456',
|
|
permissions: OC.PERMISSION_ALL
|
|
})];
|
|
pageSizeStub = sinon.stub(OCA.Files.FileList.prototype, 'pageSize').returns(20);
|
|
fileList = new OCA.Files.FileList($('#app-content-files'), {
|
|
filesClient: filesClient
|
|
});
|
|
});
|
|
afterEach(function() {
|
|
testFiles = undefined;
|
|
if (fileList) {
|
|
fileList.destroy();
|
|
}
|
|
fileList = undefined;
|
|
|
|
notificationStub.restore();
|
|
bcResizeStub.restore();
|
|
pageSizeStub.restore();
|
|
redirectStub.restore();
|
|
});
|
|
describe('Getters', function() {
|
|
it('Returns the current directory', function() {
|
|
$('#dir').val('/one/two/three');
|
|
expect(fileList.getCurrentDirectory()).toEqual('/one/two/three');
|
|
});
|
|
it('Returns the directory permissions as int', function() {
|
|
$('#permissions').val('23');
|
|
expect(fileList.getDirectoryPermissions()).toEqual(23);
|
|
});
|
|
});
|
|
describe('Adding files', function() {
|
|
var clock, now;
|
|
beforeEach(function() {
|
|
// to prevent date comparison issues
|
|
clock = sinon.useFakeTimers();
|
|
now = new Date();
|
|
});
|
|
afterEach(function() {
|
|
clock.restore();
|
|
});
|
|
it('generates file element with correct attributes when calling add() with file data', function() {
|
|
var fileData = new FileInfo({
|
|
id: 18,
|
|
name: 'testName.txt',
|
|
mimetype: 'text/plain',
|
|
size: 1234,
|
|
etag: 'a01234c',
|
|
mtime: 123456
|
|
});
|
|
var $tr = fileList.add(fileData);
|
|
|
|
expect($tr).toBeDefined();
|
|
expect($tr[0].tagName.toLowerCase()).toEqual('tr');
|
|
expect($tr.attr('data-id')).toEqual('18');
|
|
expect($tr.attr('data-type')).toEqual('file');
|
|
expect($tr.attr('data-file')).toEqual('testName.txt');
|
|
expect($tr.attr('data-size')).toEqual('1234');
|
|
expect($tr.attr('data-etag')).toEqual('a01234c');
|
|
expect($tr.attr('data-permissions')).toEqual('31');
|
|
expect($tr.attr('data-mime')).toEqual('text/plain');
|
|
expect($tr.attr('data-mtime')).toEqual('123456');
|
|
expect($tr.find('a.name').attr('href'))
|
|
.toEqual(OC.webroot + '/remote.php/webdav/subdir/testName.txt');
|
|
expect($tr.find('.nametext').text().trim()).toEqual('testName.txt');
|
|
|
|
expect($tr.find('.filesize').text()).toEqual('1 KB');
|
|
expect($tr.find('.date').text()).not.toEqual('?');
|
|
expect(fileList.findFileEl('testName.txt')[0]).toEqual($tr[0]);
|
|
});
|
|
it('generates dir element with correct attributes when calling add() with dir data', function() {
|
|
var fileData = new FileInfo({
|
|
id: 19,
|
|
name: 'testFolder',
|
|
mimetype: 'httpd/unix-directory',
|
|
size: 1234,
|
|
etag: 'a01234c',
|
|
mtime: 123456
|
|
});
|
|
var $tr = fileList.add(fileData);
|
|
|
|
expect($tr).toBeDefined();
|
|
expect($tr[0].tagName.toLowerCase()).toEqual('tr');
|
|
expect($tr.attr('data-id')).toEqual('19');
|
|
expect($tr.attr('data-type')).toEqual('dir');
|
|
expect($tr.attr('data-file')).toEqual('testFolder');
|
|
expect($tr.attr('data-size')).toEqual('1234');
|
|
expect($tr.attr('data-etag')).toEqual('a01234c');
|
|
expect($tr.attr('data-permissions')).toEqual('31');
|
|
expect($tr.attr('data-mime')).toEqual('httpd/unix-directory');
|
|
expect($tr.attr('data-mtime')).toEqual('123456');
|
|
|
|
expect($tr.find('.filesize').text()).toEqual('1 KB');
|
|
expect($tr.find('.date').text()).not.toEqual('?');
|
|
|
|
expect(fileList.findFileEl('testFolder')[0]).toEqual($tr[0]);
|
|
});
|
|
it('generates file element with default attributes when calling add() with minimal data', function() {
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'testFile.txt'
|
|
};
|
|
|
|
clock.tick(123456);
|
|
var $tr = fileList.add(fileData);
|
|
|
|
expect($tr).toBeDefined();
|
|
expect($tr[0].tagName.toLowerCase()).toEqual('tr');
|
|
expect($tr.attr('data-id')).toBeUndefined();
|
|
expect($tr.attr('data-type')).toEqual('file');
|
|
expect($tr.attr('data-file')).toEqual('testFile.txt');
|
|
expect($tr.attr('data-size')).toBeUndefined();
|
|
expect($tr.attr('data-etag')).toBeUndefined();
|
|
expect($tr.attr('data-permissions')).toEqual('31');
|
|
expect($tr.attr('data-mime')).toBeUndefined();
|
|
expect($tr.attr('data-mtime')).toEqual('123456');
|
|
|
|
expect($tr.find('.filesize').text()).toEqual('Pending');
|
|
expect($tr.find('.date').text()).not.toEqual('?');
|
|
});
|
|
it('generates dir element with default attributes when calling add() with minimal data', function() {
|
|
var fileData = {
|
|
type: 'dir',
|
|
name: 'testFolder'
|
|
};
|
|
clock.tick(123456);
|
|
var $tr = fileList.add(fileData);
|
|
|
|
expect($tr).toBeDefined();
|
|
expect($tr[0].tagName.toLowerCase()).toEqual('tr');
|
|
expect($tr.attr('data-id')).toBeUndefined();
|
|
expect($tr.attr('data-type')).toEqual('dir');
|
|
expect($tr.attr('data-file')).toEqual('testFolder');
|
|
expect($tr.attr('data-size')).toBeUndefined();
|
|
expect($tr.attr('data-etag')).toBeUndefined();
|
|
expect($tr.attr('data-permissions')).toEqual('31');
|
|
expect($tr.attr('data-mime')).toEqual('httpd/unix-directory');
|
|
expect($tr.attr('data-mtime')).toEqual('123456');
|
|
|
|
expect($tr.find('.filesize').text()).toEqual('Pending');
|
|
expect($tr.find('.date').text()).not.toEqual('?');
|
|
});
|
|
it('generates file element with zero size when size is explicitly zero', function() {
|
|
var fileData = {
|
|
type: 'dir',
|
|
name: 'testFolder',
|
|
size: '0'
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
expect($tr.find('.filesize').text()).toEqual('0 KB');
|
|
});
|
|
it('generates file element with unknown date when mtime invalid', function() {
|
|
var fileData = {
|
|
type: 'dir',
|
|
name: 'testFolder',
|
|
mtime: -1
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
expect($tr.find('.date .modified').text()).toEqual('?');
|
|
});
|
|
it('adds new file to the end of the list', function() {
|
|
var $tr;
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'ZZZ.txt'
|
|
};
|
|
fileList.setFiles(testFiles);
|
|
$tr = fileList.add(fileData);
|
|
expect($tr.index()).toEqual(4);
|
|
});
|
|
it('inserts files in a sorted manner when insert option is enabled', function() {
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
fileList.add(testFiles[i]);
|
|
}
|
|
expect(fileList.files[0].name).toEqual('somedir');
|
|
expect(fileList.files[1].name).toEqual('One.txt');
|
|
expect(fileList.files[2].name).toEqual('Three.pdf');
|
|
expect(fileList.files[3].name).toEqual('Two.jpg');
|
|
});
|
|
it('inserts new file at correct position', function() {
|
|
var $tr;
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'P comes after O.txt'
|
|
};
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
fileList.add(testFiles[i]);
|
|
}
|
|
$tr = fileList.add(fileData);
|
|
// after "One.txt"
|
|
expect($tr.index()).toEqual(2);
|
|
expect(fileList.files[2]).toEqual(fileData);
|
|
});
|
|
it('inserts new folder at correct position in insert mode', function() {
|
|
var $tr;
|
|
var fileData = {
|
|
type: 'dir',
|
|
name: 'somedir2 comes after somedir'
|
|
};
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
fileList.add(testFiles[i]);
|
|
}
|
|
$tr = fileList.add(fileData);
|
|
expect($tr.index()).toEqual(1);
|
|
expect(fileList.files[1]).toEqual(fileData);
|
|
});
|
|
it('inserts new file at the end correctly', function() {
|
|
var $tr;
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'zzz.txt'
|
|
};
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
fileList.add(testFiles[i]);
|
|
}
|
|
$tr = fileList.add(fileData);
|
|
expect($tr.index()).toEqual(4);
|
|
expect(fileList.files[4]).toEqual(fileData);
|
|
});
|
|
it('removes empty content message and shows summary when adding first file', function() {
|
|
var $summary;
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'first file.txt',
|
|
size: 12
|
|
};
|
|
fileList.setFiles([]);
|
|
expect(fileList.isEmpty).toEqual(true);
|
|
fileList.add(fileData);
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
// yes, ugly...
|
|
expect($summary.find('.info').text()).toEqual('0 folders and 1 file');
|
|
expect($summary.find('.dirinfo').hasClass('hidden')).toEqual(true);
|
|
expect($summary.find('.fileinfo').hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.filesize').text()).toEqual('12 B');
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(false);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(true);
|
|
expect(fileList.isEmpty).toEqual(false);
|
|
});
|
|
it('correctly adds the extension markup and show hidden files completely in gray', function() {
|
|
var $tr;
|
|
var testDataAndExpectedResult = [
|
|
{file: {type: 'file', name: 'ZZZ.txt'}, extension: '.txt'},
|
|
{file: {type: 'file', name: 'ZZZ.tar.gz'}, extension: '.gz'},
|
|
{file: {type: 'file', name: 'test.with.some.dots.in.it.txt'}, extension: '.txt'},
|
|
// we render hidden files completely in gray
|
|
{file: {type: 'file', name: '.test.with.some.dots.in.it.txt'}, extension: '.test.with.some.dots.in.it.txt'},
|
|
{file: {type: 'file', name: '.hidden'}, extension: '.hidden'},
|
|
];
|
|
fileList.setFiles(testFiles);
|
|
|
|
for(var i = 0; i < testDataAndExpectedResult.length; i++) {
|
|
var testSet = testDataAndExpectedResult[i];
|
|
var fileData = testSet['file'];
|
|
$tr = fileList.add(fileData);
|
|
expect($tr.find('.nametext .extension').text()).toEqual(testSet['extension']);
|
|
}
|
|
});
|
|
});
|
|
describe('Removing files from the list', function() {
|
|
it('Removes file from list when calling remove() and updates summary', function() {
|
|
var $summary;
|
|
var $removedEl;
|
|
fileList.setFiles(testFiles);
|
|
$removedEl = fileList.remove('One.txt');
|
|
expect($removedEl).toBeDefined();
|
|
expect($removedEl.attr('data-file')).toEqual('One.txt');
|
|
expect($('#fileList tr').length).toEqual(3);
|
|
expect(fileList.files.length).toEqual(3);
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual('1 folder and 2 files');
|
|
expect($summary.find('.dirinfo').hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.fileinfo').hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.filesize').text()).toEqual('69 KB');
|
|
expect(fileList.isEmpty).toEqual(false);
|
|
});
|
|
it('Shows empty content when removing last file', function() {
|
|
var $summary;
|
|
fileList.setFiles([testFiles[0]]);
|
|
fileList.remove('One.txt');
|
|
expect($('#fileList tr').length).toEqual(0);
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(true);
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(true);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(false);
|
|
expect(fileList.isEmpty).toEqual(true);
|
|
});
|
|
});
|
|
describe('Deleting files', function() {
|
|
var deferredDelete;
|
|
var deleteStub;
|
|
|
|
beforeEach(function() {
|
|
deferredDelete = $.Deferred();
|
|
deleteStub = sinon.stub(filesClient, 'remove').returns(deferredDelete.promise());
|
|
});
|
|
afterEach(function() {
|
|
deleteStub.restore();
|
|
});
|
|
|
|
function doDelete() {
|
|
// note: normally called from FileActions
|
|
fileList.do_delete(['One.txt', 'Two.jpg']);
|
|
|
|
expect(deleteStub.calledTwice).toEqual(true);
|
|
expect(deleteStub.getCall(0).args[0]).toEqual('/subdir/One.txt');
|
|
expect(deleteStub.getCall(1).args[0]).toEqual('/subdir/Two.jpg');
|
|
}
|
|
it('calls delete.php, removes the deleted entries and updates summary', function() {
|
|
var $summary;
|
|
fileList.setFiles(testFiles);
|
|
doDelete();
|
|
|
|
deferredDelete.resolve(200);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
expect(fileList.findFileEl('Two.jpg').length).toEqual(0);
|
|
expect(fileList.findFileEl('Three.pdf').length).toEqual(1);
|
|
expect(fileList.$fileList.find('tr').length).toEqual(2);
|
|
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual('1 folder and 1 file');
|
|
expect($summary.find('.dirinfo').hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.fileinfo').hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.filesize').text()).toEqual('57 KB');
|
|
expect(fileList.isEmpty).toEqual(false);
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(false);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(true);
|
|
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
});
|
|
it('shows busy state on files to be deleted', function() {
|
|
fileList.setFiles(testFiles);
|
|
doDelete();
|
|
|
|
expect(fileList.findFileEl('One.txt').hasClass('busy')).toEqual(true);
|
|
expect(fileList.findFileEl('Three.pdf').hasClass('busy')).toEqual(false);
|
|
});
|
|
it('shows busy state on all files when deleting all', function() {
|
|
fileList.setFiles(testFiles);
|
|
|
|
fileList.do_delete();
|
|
|
|
expect(fileList.$fileList.find('tr.busy').length).toEqual(4);
|
|
});
|
|
it('updates summary when deleting last file', function() {
|
|
var $summary;
|
|
fileList.setFiles([testFiles[0], testFiles[1]]);
|
|
doDelete();
|
|
|
|
deferredDelete.resolve(200);
|
|
|
|
expect(fileList.$fileList.find('tr').length).toEqual(0);
|
|
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(true);
|
|
expect(fileList.isEmpty).toEqual(true);
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(true);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(false);
|
|
});
|
|
it('bring back deleted item when delete call failed', function() {
|
|
fileList.setFiles(testFiles);
|
|
doDelete();
|
|
|
|
deferredDelete.reject(403);
|
|
|
|
// files are still in the list
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(1);
|
|
expect(fileList.findFileEl('Two.jpg').length).toEqual(1);
|
|
expect(fileList.$fileList.find('tr').length).toEqual(4);
|
|
|
|
expect(notificationStub.calledTwice).toEqual(true);
|
|
});
|
|
it('remove file from list if delete call returned 404 not found', function() {
|
|
fileList.setFiles(testFiles);
|
|
doDelete();
|
|
|
|
deferredDelete.reject(404);
|
|
|
|
// files are still in the list
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
expect(fileList.findFileEl('Two.jpg').length).toEqual(0);
|
|
expect(fileList.$fileList.find('tr').length).toEqual(2);
|
|
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
});
|
|
});
|
|
describe('Renaming files', function() {
|
|
var deferredRename;
|
|
var renameStub;
|
|
|
|
beforeEach(function() {
|
|
deferredRename = $.Deferred();
|
|
renameStub = sinon.stub(filesClient, 'move').returns(deferredRename.promise());
|
|
});
|
|
afterEach(function() {
|
|
renameStub.restore();
|
|
});
|
|
|
|
function doCancelRename() {
|
|
var $input;
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
fileList.add(testFiles[i]);
|
|
}
|
|
|
|
// trigger rename prompt
|
|
fileList.rename('One.txt');
|
|
$input = fileList.$fileList.find('input.filename');
|
|
// keep same name
|
|
$input.val('One.txt');
|
|
// trigger submit because triggering blur doesn't work in all browsers
|
|
$input.closest('form').trigger('submit');
|
|
|
|
expect(renameStub.notCalled).toEqual(true);
|
|
}
|
|
function doRename() {
|
|
var $input;
|
|
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
var file = testFiles[i];
|
|
file.path = '/some/subdir';
|
|
fileList.add(file, {silent: true});
|
|
}
|
|
|
|
// trigger rename prompt
|
|
fileList.rename('One.txt');
|
|
$input = fileList.$fileList.find('input.filename');
|
|
$input.val('Tu_after_three.txt');
|
|
// trigger submit because triggering blur doesn't work in all browsers
|
|
$input.closest('form').trigger('submit');
|
|
|
|
expect(renameStub.calledOnce).toEqual(true);
|
|
expect(renameStub.getCall(0).args[0]).toEqual('/some/subdir/One.txt');
|
|
expect(renameStub.getCall(0).args[1]).toEqual('/some/subdir/Tu_after_three.txt');
|
|
}
|
|
it('Inserts renamed file entry at correct position if rename ajax call suceeded', function() {
|
|
doRename();
|
|
|
|
deferredRename.resolve(201);
|
|
|
|
// element stays renamed
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
expect(fileList.findFileEl('Tu_after_three.txt').length).toEqual(1);
|
|
expect(fileList.findFileEl('Tu_after_three.txt').index()).toEqual(2); // after Two.jpg
|
|
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
});
|
|
it('Reverts file entry if rename ajax call failed', function() {
|
|
doRename();
|
|
|
|
deferredRename.reject(403);
|
|
|
|
// element was reverted
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(1);
|
|
expect(fileList.findFileEl('One.txt').index()).toEqual(1); // after somedir
|
|
expect(fileList.findFileEl('Tu_after_three.txt').length).toEqual(0);
|
|
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
});
|
|
it('Correctly updates file link after rename', function() {
|
|
var $tr;
|
|
doRename();
|
|
|
|
deferredRename.resolve(201);
|
|
|
|
$tr = fileList.findFileEl('Tu_after_three.txt');
|
|
expect($tr.find('a.name').attr('href'))
|
|
.toEqual(OC.webroot + '/remote.php/webdav/some/subdir/Tu_after_three.txt');
|
|
});
|
|
it('Triggers "fileActionsReady" event after rename', function() {
|
|
var handler = sinon.stub();
|
|
fileList.$fileList.on('fileActionsReady', handler);
|
|
doRename();
|
|
expect(handler.notCalled).toEqual(true);
|
|
|
|
deferredRename.resolve(201);
|
|
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(fileList.$fileList.find('.test').length).toEqual(0);
|
|
});
|
|
it('Leaves the summary alone when reinserting renamed element', function() {
|
|
var $summary = $('#filestable .summary');
|
|
doRename();
|
|
|
|
deferredRename.resolve(201);
|
|
|
|
expect($summary.find('.info').text()).toEqual('1 folder and 3 files');
|
|
});
|
|
it('Leaves the summary alone when cancel renaming', function() {
|
|
var $summary = $('#filestable .summary');
|
|
doCancelRename();
|
|
expect($summary.find('.info').text()).toEqual('1 folder and 3 files');
|
|
});
|
|
it('Shows busy state while rename in progress', function() {
|
|
var $tr;
|
|
doRename();
|
|
|
|
// element is renamed before the request finishes
|
|
$tr = fileList.findFileEl('Tu_after_three.txt');
|
|
expect($tr.length).toEqual(1);
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
// file actions are hidden
|
|
expect($tr.hasClass('busy')).toEqual(true);
|
|
|
|
// input and form are gone
|
|
expect(fileList.$fileList.find('input.filename').length).toEqual(0);
|
|
expect(fileList.$fileList.find('form').length).toEqual(0);
|
|
});
|
|
it('Validates the file name', function() {
|
|
var $input, $tr;
|
|
|
|
for (var i = 0; i < testFiles.length; i++) {
|
|
fileList.add(testFiles[i], {silent: true});
|
|
}
|
|
|
|
// trigger rename prompt
|
|
fileList.rename('One.txt');
|
|
$input = fileList.$fileList.find('input.filename');
|
|
$input.val('Two.jpg');
|
|
|
|
// simulate key to trigger validation
|
|
$input.trigger(new $.Event('keyup', {keyCode: 97}));
|
|
|
|
// input is still there with error
|
|
expect(fileList.$fileList.find('input.filename').length).toEqual(1);
|
|
expect(fileList.$fileList.find('input.filename').hasClass('error')).toEqual(true);
|
|
|
|
// trigger submit does not send server request
|
|
$input.closest('form').trigger('submit');
|
|
expect(renameStub.notCalled).toEqual(true);
|
|
|
|
// simulate escape key
|
|
$input.trigger(new $.Event('keyup', {keyCode: 27}));
|
|
|
|
// element is added back with the correct name
|
|
$tr = fileList.findFileEl('One.txt');
|
|
expect($tr.length).toEqual(1);
|
|
expect($tr.find('a .nametext').text().trim()).toEqual('One.txt');
|
|
expect($tr.find('a.name').is(':visible')).toEqual(true);
|
|
|
|
$tr = fileList.findFileEl('Two.jpg');
|
|
expect($tr.length).toEqual(1);
|
|
expect($tr.find('a .nametext').text().trim()).toEqual('Two.jpg');
|
|
expect($tr.find('a.name').is(':visible')).toEqual(true);
|
|
|
|
// input and form are gone
|
|
expect(fileList.$fileList.find('input.filename').length).toEqual(0);
|
|
expect(fileList.$fileList.find('form').length).toEqual(0);
|
|
});
|
|
it('Restores thumbnail when rename was cancelled', function() {
|
|
doRename();
|
|
|
|
expect(OC.TestUtil.getImageUrl(fileList.findFileEl('Tu_after_three.txt').find('.thumbnail')))
|
|
.toEqual(OC.imagePath('core', 'loading.gif'));
|
|
|
|
deferredRename.reject(409);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(1);
|
|
expect(OC.TestUtil.getImageUrl(fileList.findFileEl('One.txt').find('.thumbnail')))
|
|
.toEqual(OC.imagePath('core', 'filetypes/text.svg'));
|
|
});
|
|
});
|
|
describe('Moving files', function() {
|
|
var deferredMove;
|
|
var moveStub;
|
|
|
|
beforeEach(function() {
|
|
deferredMove = $.Deferred();
|
|
moveStub = sinon.stub(filesClient, 'move').returns(deferredMove.promise());
|
|
|
|
fileList.setFiles(testFiles);
|
|
});
|
|
afterEach(function() {
|
|
moveStub.restore();
|
|
});
|
|
|
|
it('Moves single file to target folder', function() {
|
|
fileList.move('One.txt', '/somedir');
|
|
|
|
expect(moveStub.calledOnce).toEqual(true);
|
|
expect(moveStub.getCall(0).args[0]).toEqual('/subdir/One.txt');
|
|
expect(moveStub.getCall(0).args[1]).toEqual('/somedir/One.txt');
|
|
|
|
deferredMove.resolve(201);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
|
|
// folder size has increased
|
|
expect(fileList.findFileEl('somedir').data('size')).toEqual(262);
|
|
expect(fileList.findFileEl('somedir').find('.filesize').text()).toEqual('262 B');
|
|
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
});
|
|
it('Moves list of files to target folder', function() {
|
|
var deferredMove1 = $.Deferred();
|
|
var deferredMove2 = $.Deferred();
|
|
moveStub.onCall(0).returns(deferredMove1.promise());
|
|
moveStub.onCall(1).returns(deferredMove2.promise());
|
|
|
|
fileList.move(['One.txt', 'Two.jpg'], '/somedir');
|
|
|
|
expect(moveStub.calledTwice).toEqual(true);
|
|
expect(moveStub.getCall(0).args[0]).toEqual('/subdir/One.txt');
|
|
expect(moveStub.getCall(0).args[1]).toEqual('/somedir/One.txt');
|
|
expect(moveStub.getCall(1).args[0]).toEqual('/subdir/Two.jpg');
|
|
expect(moveStub.getCall(1).args[1]).toEqual('/somedir/Two.jpg');
|
|
|
|
deferredMove1.resolve(201);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
|
|
// folder size has increased during move
|
|
expect(fileList.findFileEl('somedir').data('size')).toEqual(262);
|
|
expect(fileList.findFileEl('somedir').find('.filesize').text()).toEqual('262 B');
|
|
|
|
deferredMove2.resolve(201);
|
|
|
|
expect(fileList.findFileEl('Two.jpg').length).toEqual(0);
|
|
|
|
// folder size has increased
|
|
expect(fileList.findFileEl('somedir').data('size')).toEqual(12311);
|
|
expect(fileList.findFileEl('somedir').find('.filesize').text()).toEqual('12 KB');
|
|
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
});
|
|
it('Shows notification if a file could not be moved', function() {
|
|
fileList.move('One.txt', '/somedir');
|
|
|
|
expect(moveStub.calledOnce).toEqual(true);
|
|
|
|
deferredMove.reject(409);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(1);
|
|
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
expect(notificationStub.getCall(0).args[0]).toEqual('Could not move "One.txt"');
|
|
});
|
|
it('Restores thumbnail if a file could not be moved', function() {
|
|
fileList.move('One.txt', '/somedir');
|
|
|
|
expect(OC.TestUtil.getImageUrl(fileList.findFileEl('One.txt').find('.thumbnail')))
|
|
.toEqual(OC.imagePath('core', 'loading.gif'));
|
|
|
|
expect(moveStub.calledOnce).toEqual(true);
|
|
|
|
deferredMove.reject(409);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(1);
|
|
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
expect(notificationStub.getCall(0).args[0]).toEqual('Could not move "One.txt"');
|
|
|
|
expect(OC.TestUtil.getImageUrl(fileList.findFileEl('One.txt').find('.thumbnail')))
|
|
.toEqual(OC.imagePath('core', 'filetypes/text.svg'));
|
|
});
|
|
});
|
|
describe('Update file', function() {
|
|
it('does not change summary', function() {
|
|
var $summary = $('#filestable .summary');
|
|
var fileData = new FileInfo({
|
|
type: 'file',
|
|
name: 'test file',
|
|
});
|
|
var $tr = fileList.add(fileData);
|
|
|
|
expect($summary.find('.info').text()).toEqual('0 folders and 1 file');
|
|
|
|
var model = fileList.getModelForFile('test file');
|
|
model.set({size: '100'});
|
|
expect($summary.find('.info').text()).toEqual('0 folders and 1 file');
|
|
});
|
|
})
|
|
describe('List rendering', function() {
|
|
it('renders a list of files using add()', function() {
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect(fileList.files).toEqual([]);
|
|
fileList.setFiles(testFiles);
|
|
expect($('#fileList tr').length).toEqual(4);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect(fileList.files).toEqual(testFiles);
|
|
});
|
|
it('updates summary using the file sizes', function() {
|
|
var $summary;
|
|
fileList.setFiles(testFiles);
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual('1 folder and 3 files');
|
|
expect($summary.find('.filesize').text()).toEqual('69 KB');
|
|
});
|
|
it('shows headers, summary and hide empty content message after setting files', function(){
|
|
fileList.setFiles(testFiles);
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(false);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(true);
|
|
expect(fileList.$el.find('.summary').hasClass('hidden')).toEqual(false);
|
|
});
|
|
it('hides headers, summary and show empty content message after setting empty file list', function(){
|
|
fileList.setFiles([]);
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(true);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(false);
|
|
expect($('#emptycontent .uploadmessage').hasClass('hidden')).toEqual(false);
|
|
expect(fileList.$el.find('.summary').hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('hides headers, upload message, and summary when list is empty and user has no creation permission', function(){
|
|
$('#permissions').val(0);
|
|
fileList.setFiles([]);
|
|
expect($('#filestable thead th').hasClass('hidden')).toEqual(true);
|
|
expect($('#emptycontent').hasClass('hidden')).toEqual(false);
|
|
expect($('#emptycontent .uploadmessage').hasClass('hidden')).toEqual(true);
|
|
expect(fileList.$el.find('.summary').hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('calling findFileEl() can find existing file element', function() {
|
|
fileList.setFiles(testFiles);
|
|
expect(fileList.findFileEl('Two.jpg').length).toEqual(1);
|
|
});
|
|
it('calling findFileEl() returns empty when file not found in file', function() {
|
|
fileList.setFiles(testFiles);
|
|
expect(fileList.findFileEl('unexist.dat').length).toEqual(0);
|
|
});
|
|
it('only add file if in same current directory', function() {
|
|
$('#dir').val('/current dir');
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'testFile.txt',
|
|
directory: '/current dir'
|
|
};
|
|
fileList.add(fileData);
|
|
expect(fileList.findFileEl('testFile.txt').length).toEqual(1);
|
|
});
|
|
it('triggers "fileActionsReady" event after update', function() {
|
|
var handler = sinon.stub();
|
|
fileList.$fileList.on('fileActionsReady', handler);
|
|
fileList.setFiles(testFiles);
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(handler.getCall(0).args[0].$files.length).toEqual(testFiles.length);
|
|
});
|
|
it('triggers "fileActionsReady" event after single add', function() {
|
|
var handler = sinon.stub();
|
|
var $tr;
|
|
fileList.setFiles(testFiles);
|
|
fileList.$fileList.on('fileActionsReady', handler);
|
|
$tr = fileList.add({name: 'test.txt'});
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(handler.getCall(0).args[0].$files.is($tr)).toEqual(true);
|
|
});
|
|
it('triggers "fileActionsReady" event after next page load with the newly appended files', function() {
|
|
var handler = sinon.stub();
|
|
fileList.setFiles(generateFiles(0, 64));
|
|
fileList.$fileList.on('fileActionsReady', handler);
|
|
fileList._nextPage();
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(handler.getCall(0).args[0].$files.length).toEqual(fileList.pageSize());
|
|
});
|
|
it('does not trigger "fileActionsReady" event after single add with silent argument', function() {
|
|
var handler = sinon.stub();
|
|
fileList.setFiles(testFiles);
|
|
fileList.$fileList.on('fileActionsReady', handler);
|
|
fileList.add({name: 'test.txt'}, {silent: true});
|
|
expect(handler.notCalled).toEqual(true);
|
|
});
|
|
it('triggers "updated" event after update', function() {
|
|
var handler = sinon.stub();
|
|
fileList.$fileList.on('updated', handler);
|
|
fileList.setFiles(testFiles);
|
|
expect(handler.calledOnce).toEqual(true);
|
|
});
|
|
it('does not update summary when removing non-existing files', function() {
|
|
var $summary;
|
|
// single file
|
|
fileList.setFiles([testFiles[0]]);
|
|
$summary = $('#filestable .summary');
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual('0 folders and 1 file');
|
|
fileList.remove('unexist.txt');
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual('0 folders and 1 file');
|
|
});
|
|
});
|
|
describe('Filtered list rendering', function() {
|
|
it('filters the list of files using filter()', function() {
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect(fileList.files).toEqual([]);
|
|
fileList.setFiles(testFiles);
|
|
var $summary = $('#filestable .summary');
|
|
var $nofilterresults = fileList.$el.find(".nofilterresults");
|
|
expect($nofilterresults.length).toEqual(1);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(4);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
fileList.setFilter('e');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(3);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual("1 folder and 2 files match 'e'");
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
fileList.setFilter('ee');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(1);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual("0 folders and 1 file matches 'ee'");
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
fileList.setFilter('eee');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(0);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(true);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(false);
|
|
|
|
fileList.setFilter('ee');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(1);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual("0 folders and 1 file matches 'ee'");
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
fileList.setFilter('e');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(3);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual("1 folder and 2 files match 'e'");
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
fileList.setFilter('');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(4);
|
|
expect(fileList.files.length).toEqual(4);
|
|
expect($summary.hasClass('hidden')).toEqual(false);
|
|
expect($summary.find('.info').text()).toEqual("1 folder and 3 files");
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('hides the emptyfiles notice when using filter()', function() {
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect(fileList.files).toEqual([]);
|
|
fileList.setFiles([]);
|
|
var $summary = $('#filestable .summary');
|
|
var $emptycontent = fileList.$el.find("#emptycontent");
|
|
var $nofilterresults = fileList.$el.find(".nofilterresults");
|
|
expect($emptycontent.length).toEqual(1);
|
|
expect($nofilterresults.length).toEqual(1);
|
|
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(0);
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect($summary.hasClass('hidden')).toEqual(true);
|
|
expect($emptycontent.hasClass('hidden')).toEqual(false);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
fileList.setFilter('e');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(0);
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect($summary.hasClass('hidden')).toEqual(true);
|
|
expect($emptycontent.hasClass('hidden')).toEqual(true);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(false);
|
|
|
|
fileList.setFilter('');
|
|
expect($('#fileList tr:not(.hidden)').length).toEqual(0);
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect($summary.hasClass('hidden')).toEqual(true);
|
|
expect($emptycontent.hasClass('hidden')).toEqual(false);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('does not show the emptyfiles or nofilterresults notice when the mask is active', function() {
|
|
expect(fileList.files.length).toEqual(0);
|
|
expect(fileList.files).toEqual([]);
|
|
fileList.showMask();
|
|
fileList.setFiles(testFiles);
|
|
var $emptycontent = fileList.$el.find("#emptycontent");
|
|
var $nofilterresults = fileList.$el.find(".nofilterresults");
|
|
expect($emptycontent.length).toEqual(1);
|
|
expect($nofilterresults.length).toEqual(1);
|
|
|
|
expect($emptycontent.hasClass('hidden')).toEqual(true);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
|
|
/*
|
|
fileList.setFilter('e');
|
|
expect($emptycontent.hasClass('hidden')).toEqual(true);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(false);
|
|
*/
|
|
|
|
fileList.setFilter('');
|
|
expect($emptycontent.hasClass('hidden')).toEqual(true);
|
|
expect($nofilterresults.hasClass('hidden')).toEqual(true);
|
|
});
|
|
});
|
|
describe('Rendering next page on scroll', function() {
|
|
beforeEach(function() {
|
|
fileList.setFiles(generateFiles(0, 64));
|
|
});
|
|
it('renders only the first page', function() {
|
|
expect(fileList.files.length).toEqual(65);
|
|
expect($('#fileList tr').length).toEqual(20);
|
|
});
|
|
it('renders the second page when scrolling down (trigger nextPage)', function() {
|
|
// TODO: can't simulate scrolling here, so calling nextPage directly
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(40);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(60);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(65);
|
|
fileList._nextPage(true);
|
|
// stays at 65
|
|
expect($('#fileList tr').length).toEqual(65);
|
|
});
|
|
it('inserts into the DOM if insertion point is in the visible page ', function() {
|
|
fileList.add({
|
|
id: 2000,
|
|
type: 'file',
|
|
name: 'File with index 15b.txt'
|
|
});
|
|
expect($('#fileList tr').length).toEqual(21);
|
|
expect(fileList.findFileEl('File with index 15b.txt').index()).toEqual(16);
|
|
});
|
|
it('does not inserts into the DOM if insertion point is not the visible page ', function() {
|
|
fileList.add({
|
|
id: 2000,
|
|
type: 'file',
|
|
name: 'File with index 28b.txt'
|
|
});
|
|
expect($('#fileList tr').length).toEqual(20);
|
|
expect(fileList.findFileEl('File with index 28b.txt').length).toEqual(0);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(40);
|
|
expect(fileList.findFileEl('File with index 28b.txt').index()).toEqual(29);
|
|
});
|
|
it('appends into the DOM when inserting a file after the last visible element', function() {
|
|
fileList.add({
|
|
id: 2000,
|
|
type: 'file',
|
|
name: 'File with index 19b.txt'
|
|
});
|
|
expect($('#fileList tr').length).toEqual(21);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(41);
|
|
});
|
|
it('appends into the DOM when inserting a file on the last page when visible', function() {
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(40);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(60);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(65);
|
|
fileList._nextPage(true);
|
|
fileList.add({
|
|
id: 2000,
|
|
type: 'file',
|
|
name: 'File with index 88.txt'
|
|
});
|
|
expect($('#fileList tr').length).toEqual(66);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(66);
|
|
});
|
|
it('shows additional page when appending a page of files and scrolling down', function() {
|
|
var newFiles = generateFiles(66, 81);
|
|
for (var i = 0; i < newFiles.length; i++) {
|
|
fileList.add(newFiles[i]);
|
|
}
|
|
expect($('#fileList tr').length).toEqual(20);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(40);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(60);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(80);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(81);
|
|
fileList._nextPage(true);
|
|
expect($('#fileList tr').length).toEqual(81);
|
|
});
|
|
it('automatically renders next page when there are not enough elements visible', function() {
|
|
// delete the 15 first elements
|
|
for (var i = 0; i < 15; i++) {
|
|
fileList.remove(fileList.files[0].name);
|
|
}
|
|
// still makes sure that there are 20 elements visible, if any
|
|
expect($('#fileList tr').length).toEqual(25);
|
|
});
|
|
});
|
|
describe('file previews', function() {
|
|
var previewLoadStub;
|
|
|
|
beforeEach(function() {
|
|
previewLoadStub = sinon.stub(OCA.Files.FileList.prototype, 'lazyLoadPreview');
|
|
});
|
|
afterEach(function() {
|
|
previewLoadStub.restore();
|
|
});
|
|
it('renders default file icon when none provided and no mime type is set', function() {
|
|
var fileData = {
|
|
name: 'testFile.txt'
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
var $imgDiv = $tr.find('td.filename .thumbnail');
|
|
expect(OC.TestUtil.getImageUrl($imgDiv)).toEqual(OC.webroot + '/core/img/filetypes/file.svg');
|
|
// tries to load preview
|
|
expect(previewLoadStub.calledOnce).toEqual(true);
|
|
});
|
|
it('renders default icon for folder when none provided', function() {
|
|
var fileData = {
|
|
name: 'test dir',
|
|
mimetype: 'httpd/unix-directory'
|
|
};
|
|
|
|
var $tr = fileList.add(fileData);
|
|
var $imgDiv = $tr.find('td.filename .thumbnail');
|
|
expect(OC.TestUtil.getImageUrl($imgDiv)).toEqual(OC.webroot + '/core/img/filetypes/folder.svg');
|
|
// no preview since it's a directory
|
|
expect(previewLoadStub.notCalled).toEqual(true);
|
|
});
|
|
it('renders provided icon for file when provided', function() {
|
|
var fileData = new FileInfo({
|
|
type: 'file',
|
|
name: 'test file',
|
|
icon: OC.webroot + '/core/img/filetypes/application-pdf.svg',
|
|
mimetype: 'application/pdf'
|
|
});
|
|
var $tr = fileList.add(fileData);
|
|
var $imgDiv = $tr.find('td.filename .thumbnail');
|
|
expect(OC.TestUtil.getImageUrl($imgDiv)).toEqual(OC.webroot + '/core/img/filetypes/application-pdf.svg');
|
|
// try loading preview
|
|
expect(previewLoadStub.calledOnce).toEqual(true);
|
|
});
|
|
it('renders provided icon for file when provided', function() {
|
|
var fileData = new FileInfo({
|
|
name: 'somefile.pdf',
|
|
icon: OC.webroot + '/core/img/filetypes/application-pdf.svg'
|
|
});
|
|
|
|
var $tr = fileList.add(fileData);
|
|
var $imgDiv = $tr.find('td.filename .thumbnail');
|
|
expect(OC.TestUtil.getImageUrl($imgDiv)).toEqual(OC.webroot + '/core/img/filetypes/application-pdf.svg');
|
|
// try loading preview
|
|
expect(previewLoadStub.calledOnce).toEqual(true);
|
|
});
|
|
it('renders provided icon for folder when provided', function() {
|
|
var fileData = new FileInfo({
|
|
name: 'some folder',
|
|
mimetype: 'httpd/unix-directory',
|
|
icon: OC.webroot + '/core/img/filetypes/folder-alt.svg'
|
|
});
|
|
|
|
var $tr = fileList.add(fileData);
|
|
var $imgDiv = $tr.find('td.filename .thumbnail');
|
|
expect(OC.TestUtil.getImageUrl($imgDiv)).toEqual(OC.webroot + '/core/img/filetypes/folder-alt.svg');
|
|
// do not load preview for folders
|
|
expect(previewLoadStub.notCalled).toEqual(true);
|
|
});
|
|
it('renders preview when no icon was provided', function() {
|
|
var fileData = {
|
|
type: 'file',
|
|
name: 'test file'
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
var $td = $tr.find('td.filename');
|
|
expect(OC.TestUtil.getImageUrl($td.find('.thumbnail')))
|
|
.toEqual(OC.webroot + '/core/img/filetypes/file.svg');
|
|
expect(previewLoadStub.calledOnce).toEqual(true);
|
|
// third argument is callback
|
|
previewLoadStub.getCall(0).args[0].callback(OC.webroot + '/somepath.png');
|
|
expect(OC.TestUtil.getImageUrl($td.find('.thumbnail'))).toEqual(OC.webroot + '/somepath.png');
|
|
});
|
|
it('does not render preview for directories', function() {
|
|
var fileData = {
|
|
type: 'dir',
|
|
mimetype: 'httpd/unix-directory',
|
|
name: 'test dir'
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
var $td = $tr.find('td.filename');
|
|
expect(OC.TestUtil.getImageUrl($td.find('.thumbnail'))).toEqual(OC.webroot + '/core/img/filetypes/folder.svg');
|
|
expect(previewLoadStub.notCalled).toEqual(true);
|
|
});
|
|
it('render external storage icon for external storage root', function() {
|
|
var fileData = {
|
|
type: 'dir',
|
|
mimetype: 'httpd/unix-directory',
|
|
name: 'test dir',
|
|
mountType: 'external-root'
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
var $td = $tr.find('td.filename');
|
|
expect(OC.TestUtil.getImageUrl($td.find('.thumbnail'))).toEqual(OC.webroot + '/core/img/filetypes/folder-external.svg');
|
|
expect(previewLoadStub.notCalled).toEqual(true);
|
|
});
|
|
it('render external storage icon for external storage subdir', function() {
|
|
var fileData = {
|
|
type: 'dir',
|
|
mimetype: 'httpd/unix-directory',
|
|
name: 'test dir',
|
|
mountType: 'external'
|
|
};
|
|
var $tr = fileList.add(fileData);
|
|
var $td = $tr.find('td.filename');
|
|
expect(OC.TestUtil.getImageUrl($td.find('.thumbnail'))).toEqual(OC.webroot + '/core/img/filetypes/folder-external.svg');
|
|
expect(previewLoadStub.notCalled).toEqual(true);
|
|
// default icon override
|
|
expect($tr.attr('data-icon')).toEqual(OC.webroot + '/core/img/filetypes/folder-external.svg');
|
|
});
|
|
|
|
});
|
|
describe('viewer mode', function() {
|
|
it('enabling viewer mode hides files table and action buttons', function() {
|
|
fileList.setViewerMode(true);
|
|
expect($('#filestable').hasClass('hidden')).toEqual(true);
|
|
expect($('.actions').hasClass('hidden')).toEqual(true);
|
|
expect($('.notCreatable').hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('disabling viewer mode restores files table and action buttons', function() {
|
|
fileList.setViewerMode(true);
|
|
fileList.setViewerMode(false);
|
|
expect($('#filestable').hasClass('hidden')).toEqual(false);
|
|
expect($('.actions').hasClass('hidden')).toEqual(false);
|
|
expect($('.notCreatable').hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('disabling viewer mode restores files table and action buttons with correct permissions', function() {
|
|
$('#permissions').val(0);
|
|
fileList.setViewerMode(true);
|
|
fileList.setViewerMode(false);
|
|
expect($('#filestable').hasClass('hidden')).toEqual(false);
|
|
expect($('.actions').hasClass('hidden')).toEqual(true);
|
|
expect($('.notCreatable').hasClass('hidden')).toEqual(false);
|
|
});
|
|
it('toggling viewer mode triggers event', function() {
|
|
var handler = sinon.stub();
|
|
fileList.$el.on('changeViewerMode', handler);
|
|
fileList.setViewerMode(true);
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(handler.getCall(0).args[0].viewerModeEnabled).toEqual(true);
|
|
|
|
handler.reset();
|
|
fileList.setViewerMode(false);
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(handler.getCall(0).args[0].viewerModeEnabled).toEqual(false);
|
|
});
|
|
});
|
|
describe('loading file list', function() {
|
|
var deferredList;
|
|
var getFolderContentsStub;
|
|
|
|
beforeEach(function() {
|
|
deferredList = $.Deferred();
|
|
getFolderContentsStub = sinon.stub(filesClient, 'getFolderContents').returns(deferredList.promise());
|
|
});
|
|
afterEach(function() {
|
|
getFolderContentsStub.restore();
|
|
});
|
|
it('fetches file list from server and renders it when reload() is called', function() {
|
|
fileList.reload();
|
|
expect(getFolderContentsStub.calledOnce).toEqual(true);
|
|
expect(getFolderContentsStub.calledWith('/subdir')).toEqual(true);
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
expect($('#fileList tr').length).toEqual(4);
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(1);
|
|
});
|
|
it('switches dir and fetches file list when calling changeDirectory()', function() {
|
|
fileList.changeDirectory('/anothersubdir');
|
|
expect(fileList.getCurrentDirectory()).toEqual('/anothersubdir');
|
|
expect(getFolderContentsStub.calledOnce).toEqual(true);
|
|
expect(getFolderContentsStub.calledWith('/anothersubdir')).toEqual(true);
|
|
});
|
|
it('converts backslashes to slashes when calling changeDirectory()', function() {
|
|
fileList.changeDirectory('/another\\subdir');
|
|
expect(fileList.getCurrentDirectory()).toEqual('/another/subdir');
|
|
});
|
|
it('switches to root dir when current directory does not exist', function() {
|
|
fileList.changeDirectory('/unexist');
|
|
deferredList.reject(404);
|
|
expect(fileList.getCurrentDirectory()).toEqual('/');
|
|
});
|
|
it('switches to root dir when current directory is forbidden', function() {
|
|
fileList.changeDirectory('/unexist');
|
|
deferredList.reject(403);
|
|
expect(fileList.getCurrentDirectory()).toEqual('/');
|
|
});
|
|
it('switches to root dir when current directory is unavailable', function() {
|
|
fileList.changeDirectory('/unexist');
|
|
deferredList.reject(500);
|
|
expect(fileList.getCurrentDirectory()).toEqual('/');
|
|
});
|
|
it('shows mask before loading file list then hides it at the end', function() {
|
|
var showMaskStub = sinon.stub(fileList, 'showMask');
|
|
var hideMaskStub = sinon.stub(fileList, 'hideMask');
|
|
fileList.changeDirectory('/anothersubdir');
|
|
expect(showMaskStub.calledOnce).toEqual(true);
|
|
expect(hideMaskStub.calledOnce).toEqual(false);
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
expect(showMaskStub.calledOnce).toEqual(true);
|
|
expect(hideMaskStub.calledOnce).toEqual(true);
|
|
showMaskStub.restore();
|
|
hideMaskStub.restore();
|
|
});
|
|
it('triggers "changeDirectory" event when changing directory', function() {
|
|
var handler = sinon.stub();
|
|
$('#app-content-files').on('changeDirectory', handler);
|
|
fileList.changeDirectory('/somedir');
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
expect(handler.calledOnce).toEqual(true);
|
|
expect(handler.getCall(0).args[0].dir).toEqual('/somedir');
|
|
});
|
|
it('changes the directory when receiving "urlChanged" event', function() {
|
|
$('#app-content-files').trigger(new $.Event('urlChanged', {view: 'files', dir: '/somedir'}));
|
|
expect(fileList.getCurrentDirectory()).toEqual('/somedir');
|
|
});
|
|
it('refreshes breadcrumb after update', function() {
|
|
var setDirSpy = sinon.spy(fileList.breadcrumb, 'setDirectory');
|
|
fileList.changeDirectory('/anothersubdir');
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
expect(fileList.breadcrumb.setDirectory.calledOnce).toEqual(true);
|
|
expect(fileList.breadcrumb.setDirectory.calledWith('/anothersubdir')).toEqual(true);
|
|
setDirSpy.restore();
|
|
getFolderContentsStub.restore();
|
|
});
|
|
});
|
|
describe('breadcrumb events', function() {
|
|
var deferredList;
|
|
var getFolderContentsStub;
|
|
|
|
beforeEach(function() {
|
|
deferredList = $.Deferred();
|
|
getFolderContentsStub = sinon.stub(filesClient, 'getFolderContents').returns(deferredList.promise());
|
|
});
|
|
afterEach(function() {
|
|
getFolderContentsStub.restore();
|
|
});
|
|
it('clicking on root breadcrumb changes directory to root', function() {
|
|
fileList.changeDirectory('/subdir/two/three with space/four/five');
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
var changeDirStub = sinon.stub(fileList, 'changeDirectory');
|
|
fileList.breadcrumb.$el.find('.crumb:eq(0)').trigger({type: 'click', which: 1});
|
|
|
|
expect(changeDirStub.calledOnce).toEqual(true);
|
|
expect(changeDirStub.getCall(0).args[0]).toEqual('/');
|
|
changeDirStub.restore();
|
|
});
|
|
it('clicking on breadcrumb changes directory', function() {
|
|
fileList.changeDirectory('/subdir/two/three with space/four/five');
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
var changeDirStub = sinon.stub(fileList, 'changeDirectory');
|
|
fileList.breadcrumb.$el.find('.crumb:eq(3)').trigger({type: 'click', which: 1});
|
|
|
|
expect(changeDirStub.calledOnce).toEqual(true);
|
|
expect(changeDirStub.getCall(0).args[0]).toEqual('/subdir/two/three with space');
|
|
changeDirStub.restore();
|
|
});
|
|
it('dropping files on breadcrumb calls move operation', function() {
|
|
var testDir = '/subdir/two/three with space/four/five';
|
|
var moveStub = sinon.stub(filesClient, 'move').returns($.Deferred().promise());
|
|
fileList.changeDirectory(testDir);
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
var $crumb = fileList.breadcrumb.$el.find('.crumb:eq(3)');
|
|
// no idea what this is but is required by the handler
|
|
var ui = {
|
|
helper: {
|
|
find: sinon.stub()
|
|
}
|
|
};
|
|
// returns a list of tr that were dragged
|
|
ui.helper.find.returns([
|
|
$('<tr data-file="One.txt" data-dir="' + testDir + '"></tr>'),
|
|
$('<tr data-file="Two.jpg" data-dir="' + testDir + '"></tr>')
|
|
]);
|
|
// simulate drop event
|
|
fileList._onDropOnBreadCrumb(new $.Event('drop', {target: $crumb}), ui);
|
|
|
|
expect(moveStub.callCount).toEqual(2);
|
|
expect(moveStub.getCall(0).args[0]).toEqual(testDir + '/One.txt');
|
|
expect(moveStub.getCall(0).args[1]).toEqual('/subdir/two/three with space/One.txt');
|
|
expect(moveStub.getCall(1).args[0]).toEqual(testDir + '/Two.jpg');
|
|
expect(moveStub.getCall(1).args[1]).toEqual('/subdir/two/three with space/Two.jpg');
|
|
moveStub.restore();
|
|
});
|
|
it('dropping files on same dir breadcrumb does nothing', function() {
|
|
var testDir = '/subdir/two/three with space/four/five';
|
|
var moveStub = sinon.stub(filesClient, 'move').returns($.Deferred().promise());
|
|
fileList.changeDirectory(testDir);
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
var $crumb = fileList.breadcrumb.$el.find('.crumb:last');
|
|
// no idea what this is but is required by the handler
|
|
var ui = {
|
|
helper: {
|
|
find: sinon.stub()
|
|
}
|
|
};
|
|
// returns a list of tr that were dragged
|
|
ui.helper.find.returns([
|
|
$('<tr data-file="One.txt" data-dir="' + testDir + '"></tr>'),
|
|
$('<tr data-file="Two.jpg" data-dir="' + testDir + '"></tr>')
|
|
]);
|
|
// simulate drop event
|
|
fileList._onDropOnBreadCrumb(new $.Event('drop', {target: $crumb}), ui);
|
|
|
|
// no extra server request
|
|
expect(moveStub.notCalled).toEqual(true);
|
|
});
|
|
});
|
|
describe('Download Url', function() {
|
|
it('returns correct download URL for single files', function() {
|
|
expect(fileList.getDownloadUrl('some file.txt'))
|
|
.toEqual(OC.webroot + '/remote.php/webdav/subdir/some%20file.txt');
|
|
expect(fileList.getDownloadUrl('some file.txt', '/anotherpath/abc'))
|
|
.toEqual(OC.webroot + '/remote.php/webdav/anotherpath/abc/some%20file.txt');
|
|
$('#dir').val('/');
|
|
expect(fileList.getDownloadUrl('some file.txt'))
|
|
.toEqual(OC.webroot + '/remote.php/webdav/some%20file.txt');
|
|
});
|
|
it('returns correct download URL for multiple files', function() {
|
|
expect(fileList.getDownloadUrl(['a b c.txt', 'd e f.txt']))
|
|
.toEqual(OC.webroot + '/index.php/apps/files/ajax/download.php?dir=%2Fsubdir&files=%5B%22a%20b%20c.txt%22%2C%22d%20e%20f.txt%22%5D');
|
|
});
|
|
it('returns the correct ajax URL', function() {
|
|
expect(fileList.getAjaxUrl('test', {a:1, b:'x y'}))
|
|
.toEqual(OC.webroot + '/index.php/apps/files/ajax/test.php?a=1&b=x%20y');
|
|
});
|
|
});
|
|
describe('File selection', function() {
|
|
beforeEach(function() {
|
|
fileList.setFiles(testFiles);
|
|
});
|
|
it('Selects a file when clicking its checkbox', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(false);
|
|
$tr.find('td.filename input:checkbox').click();
|
|
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(true);
|
|
});
|
|
it('Selects/deselect a file when clicking on the name while holding Ctrl', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
var $tr2 = fileList.findFileEl('Three.pdf');
|
|
var e;
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(false);
|
|
expect($tr2.find('input:checkbox').prop('checked')).toEqual(false);
|
|
e = new $.Event('click');
|
|
e.ctrlKey = true;
|
|
$tr.find('td.filename .name').trigger(e);
|
|
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect($tr2.find('input:checkbox').prop('checked')).toEqual(false);
|
|
|
|
// click on second entry, does not clear the selection
|
|
e = new $.Event('click');
|
|
e.ctrlKey = true;
|
|
$tr2.find('td.filename .name').trigger(e);
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect($tr2.find('input:checkbox').prop('checked')).toEqual(true);
|
|
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name')).toEqual(['One.txt', 'Three.pdf']);
|
|
|
|
// deselect now
|
|
e = new $.Event('click');
|
|
e.ctrlKey = true;
|
|
$tr2.find('td.filename .name').trigger(e);
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect($tr2.find('input:checkbox').prop('checked')).toEqual(false);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name')).toEqual(['One.txt']);
|
|
});
|
|
it('Selects a range when clicking on one file then Shift clicking on another one', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
var $tr2 = fileList.findFileEl('Three.pdf');
|
|
var e;
|
|
$tr.find('td.filename input:checkbox').click();
|
|
e = new $.Event('click');
|
|
e.shiftKey = true;
|
|
$tr2.find('td.filename .name').trigger(e);
|
|
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect($tr2.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect(fileList.findFileEl('Two.jpg').find('input:checkbox').prop('checked')).toEqual(true);
|
|
var selection = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
expect(selection.length).toEqual(3);
|
|
expect(selection).toContain('One.txt');
|
|
expect(selection).toContain('Two.jpg');
|
|
expect(selection).toContain('Three.pdf');
|
|
});
|
|
it('Selects a range when clicking on one file then Shift clicking on another one that is above the first one', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
var $tr2 = fileList.findFileEl('Three.pdf');
|
|
var e;
|
|
$tr2.find('td.filename input:checkbox').click();
|
|
e = new $.Event('click');
|
|
e.shiftKey = true;
|
|
$tr.find('td.filename .name').trigger(e);
|
|
|
|
expect($tr.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect($tr2.find('input:checkbox').prop('checked')).toEqual(true);
|
|
expect(fileList.findFileEl('Two.jpg').find('input:checkbox').prop('checked')).toEqual(true);
|
|
var selection = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
expect(selection.length).toEqual(3);
|
|
expect(selection).toContain('One.txt');
|
|
expect(selection).toContain('Two.jpg');
|
|
expect(selection).toContain('Three.pdf');
|
|
});
|
|
it('Selecting all files will automatically check "select all" checkbox', function() {
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
$('#fileList tr td.filename input:checkbox').click();
|
|
expect($('.select-all').prop('checked')).toEqual(true);
|
|
});
|
|
it('Selecting all files on the first visible page will not automatically check "select all" checkbox', function() {
|
|
fileList.setFiles(generateFiles(0, 41));
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
$('#fileList tr td.filename input:checkbox').click();
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
});
|
|
it('Clicking "select all" will select/deselect all files', function() {
|
|
fileList.setFiles(generateFiles(0, 41));
|
|
$('.select-all').click();
|
|
expect($('.select-all').prop('checked')).toEqual(true);
|
|
$('#fileList tr input:checkbox').each(function() {
|
|
expect($(this).prop('checked')).toEqual(true);
|
|
});
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(42);
|
|
|
|
$('.select-all').click();
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
|
|
$('#fileList tr input:checkbox').each(function() {
|
|
expect($(this).prop('checked')).toEqual(false);
|
|
});
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(0);
|
|
});
|
|
it('Clicking "select all" then deselecting a file will uncheck "select all"', function() {
|
|
$('.select-all').click();
|
|
expect($('.select-all').prop('checked')).toEqual(true);
|
|
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('input:checkbox').click();
|
|
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(3);
|
|
});
|
|
it('Updates the selection summary when doing a few manipulations with "Select all"', function() {
|
|
$('.select-all').click();
|
|
expect($('.select-all').prop('checked')).toEqual(true);
|
|
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
// unselect one
|
|
$tr.find('input:checkbox').click();
|
|
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(3);
|
|
|
|
// select all
|
|
$('.select-all').click();
|
|
expect($('.select-all').prop('checked')).toEqual(true);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(4);
|
|
|
|
// unselect one
|
|
$tr.find('input:checkbox').click();
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(3);
|
|
|
|
// re-select it
|
|
$tr.find('input:checkbox').click();
|
|
expect($('.select-all').prop('checked')).toEqual(true);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(4);
|
|
});
|
|
it('Auto-selects files on next page when "select all" is checked', function() {
|
|
fileList.setFiles(generateFiles(0, 41));
|
|
$('.select-all').click();
|
|
|
|
expect(fileList.$fileList.find('tr input:checkbox:checked').length).toEqual(20);
|
|
fileList._nextPage(true);
|
|
expect(fileList.$fileList.find('tr input:checkbox:checked').length).toEqual(40);
|
|
fileList._nextPage(true);
|
|
expect(fileList.$fileList.find('tr input:checkbox:checked').length).toEqual(42);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name').length).toEqual(42);
|
|
});
|
|
it('Selecting files updates selection summary', function() {
|
|
var $summary = $('#headerName a.name>span:first');
|
|
expect($summary.text()).toEqual('Name');
|
|
fileList.findFileEl('One.txt').find('input:checkbox').click();
|
|
fileList.findFileEl('Three.pdf').find('input:checkbox').click();
|
|
fileList.findFileEl('somedir').find('input:checkbox').click();
|
|
expect($summary.text()).toEqual('1 folder and 2 files');
|
|
});
|
|
it('Unselecting files hides selection summary', function() {
|
|
var $summary = $('#headerName a.name>span:first');
|
|
fileList.findFileEl('One.txt').find('input:checkbox').click().click();
|
|
expect($summary.text()).toEqual('Name');
|
|
});
|
|
it('Select/deselect files shows/hides file actions', function() {
|
|
var $actions = $('#headerName .selectedActions');
|
|
var $checkbox = fileList.findFileEl('One.txt').find('input:checkbox');
|
|
expect($actions.hasClass('hidden')).toEqual(true);
|
|
$checkbox.click();
|
|
expect($actions.hasClass('hidden')).toEqual(false);
|
|
$checkbox.click();
|
|
expect($actions.hasClass('hidden')).toEqual(true);
|
|
});
|
|
it('Selection is cleared when switching dirs', function() {
|
|
$('.select-all').click();
|
|
var deferredList = $.Deferred();
|
|
var getFolderContentsStub = sinon.stub(filesClient, 'getFolderContents').returns(deferredList.promise());
|
|
|
|
fileList.changeDirectory('/');
|
|
|
|
deferredList.resolve(200, [testRoot].concat(testFiles));
|
|
|
|
expect($('.select-all').prop('checked')).toEqual(false);
|
|
expect(_.pluck(fileList.getSelectedFiles(), 'name')).toEqual([]);
|
|
|
|
getFolderContentsStub.restore();
|
|
});
|
|
it('getSelectedFiles returns the selected files even when they are on the next page', function() {
|
|
var selectedFiles;
|
|
fileList.setFiles(generateFiles(0, 41));
|
|
$('.select-all').click();
|
|
// unselect one to not have the "allFiles" case
|
|
fileList.$fileList.find('tr input:checkbox:first').click();
|
|
|
|
// only 20 files visible, must still return all the selected ones
|
|
selectedFiles = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
|
|
expect(selectedFiles.length).toEqual(41);
|
|
});
|
|
describe('clearing the selection', function() {
|
|
it('clears selected files selected individually calling setFiles()', function() {
|
|
var selectedFiles;
|
|
|
|
fileList.setFiles(generateFiles(0, 41));
|
|
fileList.$fileList.find('tr:eq(5) input:checkbox:first').click();
|
|
fileList.$fileList.find('tr:eq(7) input:checkbox:first').click();
|
|
|
|
selectedFiles = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
expect(selectedFiles.length).toEqual(2);
|
|
|
|
fileList.setFiles(generateFiles(0, 2));
|
|
|
|
selectedFiles = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
expect(selectedFiles.length).toEqual(0);
|
|
});
|
|
it('clears selected files selected with select all when calling setFiles()', function() {
|
|
var selectedFiles;
|
|
|
|
fileList.setFiles(generateFiles(0, 41));
|
|
$('.select-all').click();
|
|
|
|
selectedFiles = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
expect(selectedFiles.length).toEqual(42);
|
|
|
|
fileList.setFiles(generateFiles(0, 2));
|
|
|
|
selectedFiles = _.pluck(fileList.getSelectedFiles(), 'name');
|
|
expect(selectedFiles.length).toEqual(0);
|
|
});
|
|
});
|
|
describe('Selection overlay', function() {
|
|
it('show doesnt show the delete action if one or more files are not deletable', function () {
|
|
fileList.setFiles(testFiles);
|
|
$('#permissions').val(OC.PERMISSION_READ | OC.PERMISSION_DELETE);
|
|
$('.select-all').click();
|
|
expect(fileList.$el.find('.delete-selected').hasClass('hidden')).toEqual(false);
|
|
testFiles[0].permissions = OC.PERMISSION_READ;
|
|
$('.select-all').click();
|
|
fileList.setFiles(testFiles);
|
|
$('.select-all').click();
|
|
expect(fileList.$el.find('.delete-selected').hasClass('hidden')).toEqual(true);
|
|
});
|
|
});
|
|
describe('Actions', function() {
|
|
beforeEach(function() {
|
|
fileList.findFileEl('One.txt').find('input:checkbox').click();
|
|
fileList.findFileEl('Three.pdf').find('input:checkbox').click();
|
|
fileList.findFileEl('somedir').find('input:checkbox').click();
|
|
});
|
|
it('getSelectedFiles returns the selected file data', function() {
|
|
var files = fileList.getSelectedFiles();
|
|
expect(files.length).toEqual(3);
|
|
expect(files[0]).toEqual({
|
|
id: 1,
|
|
name: 'One.txt',
|
|
mimetype: 'text/plain',
|
|
mtime: 123456789,
|
|
type: 'file',
|
|
size: 12,
|
|
etag: 'abc',
|
|
permissions: OC.PERMISSION_ALL
|
|
});
|
|
expect(files[1]).toEqual({
|
|
id: 3,
|
|
type: 'file',
|
|
name: 'Three.pdf',
|
|
mimetype: 'application/pdf',
|
|
mtime: 234560000,
|
|
size: 58009,
|
|
etag: '123',
|
|
permissions: OC.PERMISSION_ALL
|
|
});
|
|
expect(files[2]).toEqual({
|
|
id: 4,
|
|
type: 'dir',
|
|
name: 'somedir',
|
|
mimetype: 'httpd/unix-directory',
|
|
mtime: 134560000,
|
|
size: 250,
|
|
etag: '456',
|
|
permissions: OC.PERMISSION_ALL
|
|
});
|
|
expect(files[0].id).toEqual(1);
|
|
expect(files[0].name).toEqual('One.txt');
|
|
expect(files[1].id).toEqual(3);
|
|
expect(files[1].name).toEqual('Three.pdf');
|
|
expect(files[2].id).toEqual(4);
|
|
expect(files[2].name).toEqual('somedir');
|
|
});
|
|
it('Removing a file removes it from the selection', function() {
|
|
fileList.remove('Three.pdf');
|
|
var files = fileList.getSelectedFiles();
|
|
expect(files.length).toEqual(2);
|
|
expect(files[0]).toEqual({
|
|
id: 1,
|
|
name: 'One.txt',
|
|
mimetype: 'text/plain',
|
|
mtime: 123456789,
|
|
type: 'file',
|
|
size: 12,
|
|
etag: 'abc',
|
|
permissions: OC.PERMISSION_ALL
|
|
});
|
|
expect(files[1]).toEqual({
|
|
id: 4,
|
|
type: 'dir',
|
|
name: 'somedir',
|
|
mimetype: 'httpd/unix-directory',
|
|
mtime: 134560000,
|
|
size: 250,
|
|
etag: '456',
|
|
permissions: OC.PERMISSION_ALL
|
|
});
|
|
});
|
|
describe('Download', function() {
|
|
it('Opens download URL when clicking "Download"', function() {
|
|
$('.selectedActions .download').click();
|
|
expect(redirectStub.calledOnce).toEqual(true);
|
|
expect(redirectStub.getCall(0).args[0]).toContain(OC.webroot + '/index.php/apps/files/ajax/download.php?dir=%2Fsubdir&files=%5B%22One.txt%22%2C%22Three.pdf%22%2C%22somedir%22%5D');
|
|
redirectStub.restore();
|
|
});
|
|
it('Downloads root folder when all selected in root folder', function() {
|
|
$('#dir').val('/');
|
|
$('.select-all').click();
|
|
$('.selectedActions .download').click();
|
|
expect(redirectStub.calledOnce).toEqual(true);
|
|
expect(redirectStub.getCall(0).args[0]).toContain(OC.webroot + '/index.php/apps/files/ajax/download.php?dir=%2F&files=');
|
|
});
|
|
it('Downloads parent folder when all selected in subfolder', function() {
|
|
$('.select-all').click();
|
|
$('.selectedActions .download').click();
|
|
expect(redirectStub.calledOnce).toEqual(true);
|
|
expect(redirectStub.getCall(0).args[0]).toContain(OC.webroot + '/index.php/apps/files/ajax/download.php?dir=%2F&files=subdir');
|
|
});
|
|
});
|
|
describe('Delete', function() {
|
|
var deleteStub, deferredDelete;
|
|
beforeEach(function() {
|
|
deferredDelete = $.Deferred();
|
|
deleteStub = sinon.stub(filesClient, 'remove').returns(deferredDelete.promise());
|
|
});
|
|
afterEach(function() {
|
|
deleteStub.restore();
|
|
});
|
|
it('Deletes selected files when "Delete" clicked', function() {
|
|
$('.selectedActions .delete-selected').click();
|
|
|
|
expect(deleteStub.callCount).toEqual(3);
|
|
expect(deleteStub.getCall(0).args[0]).toEqual('/subdir/One.txt');
|
|
expect(deleteStub.getCall(1).args[0]).toEqual('/subdir/Three.pdf');
|
|
expect(deleteStub.getCall(2).args[0]).toEqual('/subdir/somedir');
|
|
|
|
deferredDelete.resolve(204);
|
|
|
|
expect(fileList.findFileEl('One.txt').length).toEqual(0);
|
|
expect(fileList.findFileEl('Three.pdf').length).toEqual(0);
|
|
expect(fileList.findFileEl('somedir').length).toEqual(0);
|
|
expect(fileList.findFileEl('Two.jpg').length).toEqual(1);
|
|
});
|
|
it('Deletes all files when all selected when "Delete" clicked', function() {
|
|
$('.select-all').click();
|
|
$('.selectedActions .delete-selected').click();
|
|
|
|
expect(deleteStub.callCount).toEqual(4);
|
|
expect(deleteStub.getCall(0).args[0]).toEqual('/subdir/One.txt');
|
|
expect(deleteStub.getCall(1).args[0]).toEqual('/subdir/Two.jpg');
|
|
expect(deleteStub.getCall(2).args[0]).toEqual('/subdir/Three.pdf');
|
|
expect(deleteStub.getCall(3).args[0]).toEqual('/subdir/somedir');
|
|
|
|
deferredDelete.resolve(204);
|
|
|
|
expect(fileList.isEmpty).toEqual(true);
|
|
});
|
|
});
|
|
});
|
|
it('resets the file selection on reload', function() {
|
|
fileList.$el.find('.select-all').click();
|
|
fileList.reload();
|
|
expect(fileList.$el.find('.select-all').prop('checked')).toEqual(false);
|
|
expect(fileList.getSelectedFiles()).toEqual([]);
|
|
});
|
|
describe('Disabled selection', function() {
|
|
beforeEach(function() {
|
|
fileList._allowSelection = false;
|
|
fileList.setFiles(testFiles);
|
|
});
|
|
it('Does not render checkboxes', function() {
|
|
expect(fileList.$fileList.find('.selectCheckBox').length).toEqual(0);
|
|
});
|
|
it('Does not select a file with Ctrl or Shift if selection is not allowed', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
var $tr2 = fileList.findFileEl('Three.pdf');
|
|
var e;
|
|
e = new $.Event('click');
|
|
e.ctrlKey = true;
|
|
$tr.find('td.filename .name').trigger(e);
|
|
|
|
// click on second entry, does not clear the selection
|
|
e = new $.Event('click');
|
|
e.ctrlKey = true;
|
|
$tr2.find('td.filename .name').trigger(e);
|
|
|
|
expect(fileList.getSelectedFiles().length).toEqual(0);
|
|
|
|
// deselect now
|
|
e = new $.Event('click');
|
|
e.shiftKey = true;
|
|
$tr2.find('td.filename .name').trigger(e);
|
|
expect(fileList.getSelectedFiles().length).toEqual(0);
|
|
});
|
|
});
|
|
});
|
|
describe('Details sidebar', function() {
|
|
beforeEach(function() {
|
|
fileList.setFiles(testFiles);
|
|
fileList.showDetailsView('Two.jpg');
|
|
});
|
|
describe('registering', function() {
|
|
var addTabStub;
|
|
var addDetailStub;
|
|
|
|
beforeEach(function() {
|
|
addTabStub = sinon.stub(OCA.Files.DetailsView.prototype, 'addTabView');
|
|
addDetailStub = sinon.stub(OCA.Files.DetailsView.prototype, 'addDetailView');
|
|
});
|
|
afterEach(function() {
|
|
addTabStub.restore();
|
|
addDetailStub.restore();
|
|
});
|
|
it('forward the registered views to the underlying DetailsView', function() {
|
|
fileList.destroy();
|
|
fileList = new OCA.Files.FileList($('#app-content-files'), {
|
|
detailsViewEnabled: true
|
|
});
|
|
fileList.registerTabView(new OCA.Files.DetailTabView());
|
|
fileList.registerDetailView(new OCA.Files.DetailFileInfoView());
|
|
|
|
expect(addTabStub.calledOnce).toEqual(true);
|
|
// twice because the filelist already registers one by default
|
|
expect(addDetailStub.calledTwice).toEqual(true);
|
|
});
|
|
it('does not error when registering panels when not details view configured', function() {
|
|
fileList.destroy();
|
|
fileList = new OCA.Files.FileList($('#app-content-files'), {
|
|
detailsViewEnabled: false
|
|
});
|
|
fileList.registerTabView(new OCA.Files.DetailTabView());
|
|
fileList.registerDetailView(new OCA.Files.DetailFileInfoView());
|
|
|
|
expect(addTabStub.notCalled).toEqual(true);
|
|
expect(addDetailStub.notCalled).toEqual(true);
|
|
});
|
|
});
|
|
it('triggers file action when clicking on row if no details view configured', function() {
|
|
fileList.destroy();
|
|
fileList = new OCA.Files.FileList($('#app-content-files'), {
|
|
detailsViewEnabled: false
|
|
});
|
|
var updateDetailsViewStub = sinon.stub(fileList, '_updateDetailsView');
|
|
var actionStub = sinon.stub();
|
|
fileList.setFiles(testFiles);
|
|
fileList.fileActions.register(
|
|
'text/plain',
|
|
'Test',
|
|
OC.PERMISSION_ALL,
|
|
function() {
|
|
// Specify icon for hitory button
|
|
return OC.imagePath('core','actions/history');
|
|
},
|
|
actionStub
|
|
);
|
|
fileList.fileActions.setDefault('text/plain', 'Test');
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename>a.name').click();
|
|
expect(actionStub.calledOnce).toEqual(true);
|
|
expect(updateDetailsViewStub.notCalled).toEqual(true);
|
|
updateDetailsViewStub.restore();
|
|
});
|
|
it('highlights current file when clicked and updates sidebar', function() {
|
|
fileList.fileActions.setDefault('text/plain', 'Test');
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename>a.name').click();
|
|
expect($tr.hasClass('highlighted')).toEqual(true);
|
|
|
|
expect(fileList._detailsView.getFileInfo().id).toEqual(1);
|
|
});
|
|
it('keeps the last highlighted file when clicking outside', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename>a.name').click();
|
|
|
|
fileList.$el.find('tfoot').click();
|
|
|
|
expect($tr.hasClass('highlighted')).toEqual(true);
|
|
expect(fileList._detailsView.getFileInfo().id).toEqual(1);
|
|
});
|
|
it('removes last highlighted file when selecting via checkbox', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
|
|
// select
|
|
$tr.find('td.filename>a.name').click();
|
|
$tr.find('input:checkbox').click();
|
|
expect($tr.hasClass('highlighted')).toEqual(false);
|
|
|
|
// deselect
|
|
$tr.find('td.filename>a.name').click();
|
|
$tr.find('input:checkbox').click();
|
|
expect($tr.hasClass('highlighted')).toEqual(false);
|
|
|
|
expect(fileList._detailsView.getFileInfo()).toEqual(null);
|
|
});
|
|
it('removes last highlighted file when selecting all files via checkbox', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
|
|
// select
|
|
$tr.find('td.filename>a.name').click();
|
|
fileList.$el.find('.select-all.checkbox').click();
|
|
expect($tr.hasClass('highlighted')).toEqual(false);
|
|
|
|
// deselect
|
|
$tr.find('td.filename>a.name').click();
|
|
fileList.$el.find('.select-all.checkbox').click();
|
|
expect($tr.hasClass('highlighted')).toEqual(false);
|
|
|
|
expect(fileList._detailsView.getFileInfo()).toEqual(null);
|
|
});
|
|
it('closes sidebar whenever the currently highlighted file was removed from the list', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename>a.name').click();
|
|
expect($tr.hasClass('highlighted')).toEqual(true);
|
|
|
|
expect(fileList._detailsView.getFileInfo().id).toEqual(1);
|
|
|
|
expect($('#app-sidebar').hasClass('disappear')).toEqual(false);
|
|
fileList.remove('One.txt');
|
|
expect($('#app-sidebar').hasClass('disappear')).toEqual(true);
|
|
});
|
|
it('returns the currently selected model instance when calling getModelForFile', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename>a.name').click();
|
|
|
|
var model1 = fileList.getModelForFile('One.txt');
|
|
var model2 = fileList.getModelForFile('One.txt');
|
|
model1.set('test', true);
|
|
|
|
// it's the same model
|
|
expect(model2).toEqual(model1);
|
|
|
|
var model3 = fileList.getModelForFile($tr);
|
|
expect(model3).toEqual(model1);
|
|
});
|
|
it('closes the sidebar when switching folders', function() {
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename>a.name').click();
|
|
|
|
expect($('#app-sidebar').hasClass('disappear')).toEqual(false);
|
|
fileList.changeDirectory('/another');
|
|
expect($('#app-sidebar').hasClass('disappear')).toEqual(true);
|
|
});
|
|
});
|
|
describe('File actions', function() {
|
|
it('Clicking on a file name will trigger default action', function() {
|
|
var actionStub = sinon.stub();
|
|
fileList.setFiles(testFiles);
|
|
fileList.fileActions.registerAction({
|
|
mime: 'text/plain',
|
|
name: 'Test',
|
|
type: OCA.Files.FileActions.TYPE_INLINE,
|
|
permissions: OC.PERMISSION_ALL,
|
|
icon: function() {
|
|
// Specify icon for hitory button
|
|
return OC.imagePath('core','actions/history');
|
|
},
|
|
actionHandler: actionStub
|
|
});
|
|
fileList.fileActions.setDefault('text/plain', 'Test');
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
$tr.find('td.filename .nametext').click();
|
|
expect(actionStub.calledOnce).toEqual(true);
|
|
expect(actionStub.getCall(0).args[0]).toEqual('One.txt');
|
|
var context = actionStub.getCall(0).args[1];
|
|
expect(context.$file.is($tr)).toEqual(true);
|
|
expect(context.fileList).toBeDefined();
|
|
expect(context.fileActions).toBeDefined();
|
|
expect(context.dir).toEqual('/subdir');
|
|
});
|
|
it('redisplays actions when new actions have been registered', function() {
|
|
var actionStub = sinon.stub();
|
|
var readyHandler = sinon.stub();
|
|
var clock = sinon.useFakeTimers();
|
|
var debounceStub = sinon.stub(_, 'debounce', function(callback) {
|
|
return function() {
|
|
// defer instead of debounce, to make it work with clock
|
|
_.defer(callback);
|
|
};
|
|
});
|
|
|
|
// need to reinit the list to make the debounce call
|
|
fileList.destroy();
|
|
fileList = new OCA.Files.FileList($('#app-content-files'));
|
|
|
|
fileList.setFiles(testFiles);
|
|
|
|
fileList.$fileList.on('fileActionsReady', readyHandler);
|
|
|
|
fileList.fileActions.registerAction({
|
|
mime: 'text/plain',
|
|
name: 'Test',
|
|
type: OCA.Files.FileActions.TYPE_INLINE,
|
|
permissions: OC.PERMISSION_ALL,
|
|
icon: function() {
|
|
// Specify icon for hitory button
|
|
return OC.imagePath('core','actions/history');
|
|
},
|
|
actionHandler: actionStub
|
|
});
|
|
var $tr = fileList.findFileEl('One.txt');
|
|
expect($tr.find('.action-test').length).toEqual(0);
|
|
expect(readyHandler.notCalled).toEqual(true);
|
|
|
|
// update is delayed
|
|
clock.tick(100);
|
|
expect($tr.find('.action-test').length).toEqual(1);
|
|
expect(readyHandler.calledOnce).toEqual(true);
|
|
|
|
clock.restore();
|
|
debounceStub.restore();
|
|
});
|
|
});
|
|
describe('Sorting files', function() {
|
|
it('Toggles the sort indicator when clicking on a column header', function() {
|
|
var ASC_CLASS = fileList.SORT_INDICATOR_ASC_CLASS;
|
|
var DESC_CLASS = fileList.SORT_INDICATOR_DESC_CLASS;
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
// moves triangle to size column, check indicator on name is hidden
|
|
expect(
|
|
fileList.$el.find('.column-name .sort-indicator').hasClass('hidden')
|
|
).toEqual(true);
|
|
// check indicator on size is visible and defaults to descending
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass('hidden')
|
|
).toEqual(false);
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass(DESC_CLASS)
|
|
).toEqual(true);
|
|
|
|
// click again on size column, reverses direction
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass('hidden')
|
|
).toEqual(false);
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass(ASC_CLASS)
|
|
).toEqual(true);
|
|
|
|
// click again on size column, reverses direction
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass('hidden')
|
|
).toEqual(false);
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass(DESC_CLASS)
|
|
).toEqual(true);
|
|
|
|
// click on mtime column, moves indicator there
|
|
fileList.$el.find('.column-mtime .columntitle').click();
|
|
expect(
|
|
fileList.$el.find('.column-size .sort-indicator').hasClass('hidden')
|
|
).toEqual(true);
|
|
expect(
|
|
fileList.$el.find('.column-mtime .sort-indicator').hasClass('hidden')
|
|
).toEqual(false);
|
|
expect(
|
|
fileList.$el.find('.column-mtime .sort-indicator').hasClass(DESC_CLASS)
|
|
).toEqual(true);
|
|
});
|
|
it('Uses correct sort comparator when inserting files', function() {
|
|
testFiles.sort(OCA.Files.FileList.Comparators.size);
|
|
testFiles.reverse(); //default is descending
|
|
fileList.setFiles(testFiles);
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
var newFileData = new FileInfo({
|
|
id: 999,
|
|
name: 'new file.txt',
|
|
mimetype: 'text/plain',
|
|
size: 40001,
|
|
etag: '999'
|
|
});
|
|
fileList.add(newFileData);
|
|
expect(fileList.findFileEl('Three.pdf').index()).toEqual(0);
|
|
expect(fileList.findFileEl('new file.txt').index()).toEqual(1);
|
|
expect(fileList.findFileEl('Two.jpg').index()).toEqual(2);
|
|
expect(fileList.findFileEl('somedir').index()).toEqual(3);
|
|
expect(fileList.findFileEl('One.txt').index()).toEqual(4);
|
|
expect(fileList.files.length).toEqual(5);
|
|
expect(fileList.$fileList.find('tr').length).toEqual(5);
|
|
});
|
|
it('Uses correct reversed sort comparator when inserting files', function() {
|
|
testFiles.sort(OCA.Files.FileList.Comparators.size);
|
|
fileList.setFiles(testFiles);
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
|
|
// reverse sort
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
var newFileData = new FileInfo({
|
|
id: 999,
|
|
name: 'new file.txt',
|
|
mimetype: 'text/plain',
|
|
size: 40001,
|
|
etag: '999'
|
|
});
|
|
fileList.add(newFileData);
|
|
expect(fileList.findFileEl('One.txt').index()).toEqual(0);
|
|
expect(fileList.findFileEl('somedir').index()).toEqual(1);
|
|
expect(fileList.findFileEl('Two.jpg').index()).toEqual(2);
|
|
expect(fileList.findFileEl('new file.txt').index()).toEqual(3);
|
|
expect(fileList.findFileEl('Three.pdf').index()).toEqual(4);
|
|
expect(fileList.files.length).toEqual(5);
|
|
expect(fileList.$fileList.find('tr').length).toEqual(5);
|
|
});
|
|
it('does not sort when clicking on header whenever multiselect is enabled', function() {
|
|
var sortStub = sinon.stub(OCA.Files.FileList.prototype, 'setSort');
|
|
|
|
fileList.setFiles(testFiles);
|
|
fileList.findFileEl('One.txt').find('input:checkbox:first').click();
|
|
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
|
|
expect(sortStub.notCalled).toEqual(true);
|
|
|
|
// can sort again after deselecting
|
|
fileList.findFileEl('One.txt').find('input:checkbox:first').click();
|
|
|
|
fileList.$el.find('.column-size .columntitle').click();
|
|
|
|
expect(sortStub.calledOnce).toEqual(true);
|
|
|
|
sortStub.restore();
|
|
});
|
|
});
|
|
describe('create file', function() {
|
|
var deferredCreate;
|
|
var deferredInfo;
|
|
var createStub;
|
|
var getFileInfoStub;
|
|
|
|
beforeEach(function() {
|
|
deferredCreate = $.Deferred();
|
|
deferredInfo = $.Deferred();
|
|
createStub = sinon.stub(filesClient, 'putFileContents')
|
|
.returns(deferredCreate.promise());
|
|
getFileInfoStub = sinon.stub(filesClient, 'getFileInfo')
|
|
.returns(deferredInfo.promise());
|
|
});
|
|
afterEach(function() {
|
|
createStub.restore();
|
|
getFileInfoStub.restore();
|
|
});
|
|
|
|
it('creates file with given name and adds it to the list', function() {
|
|
fileList.createFile('test.txt');
|
|
|
|
expect(createStub.calledOnce).toEqual(true);
|
|
expect(createStub.getCall(0).args[0]).toEqual('/subdir/test.txt');
|
|
expect(createStub.getCall(0).args[2]).toEqual({
|
|
contentType: 'text/plain',
|
|
overwrite: true
|
|
});
|
|
|
|
deferredCreate.resolve(200);
|
|
|
|
expect(getFileInfoStub.calledOnce).toEqual(true);
|
|
expect(getFileInfoStub.getCall(0).args[0]).toEqual('/subdir/test.txt');
|
|
|
|
deferredInfo.resolve(
|
|
200,
|
|
new FileInfo({
|
|
path: '/subdir',
|
|
name: 'test.txt',
|
|
mimetype: 'text/plain'
|
|
})
|
|
);
|
|
|
|
var $tr = fileList.findFileEl('test.txt');
|
|
expect($tr.length).toEqual(1);
|
|
expect($tr.attr('data-mime')).toEqual('text/plain');
|
|
});
|
|
// TODO: error cases
|
|
// TODO: unique name cases
|
|
});
|
|
describe('create folder', function() {
|
|
var deferredCreate;
|
|
var deferredInfo;
|
|
var createStub;
|
|
var getFileInfoStub;
|
|
|
|
beforeEach(function() {
|
|
deferredCreate = $.Deferred();
|
|
deferredInfo = $.Deferred();
|
|
createStub = sinon.stub(filesClient, 'createDirectory')
|
|
.returns(deferredCreate.promise());
|
|
getFileInfoStub = sinon.stub(filesClient, 'getFileInfo')
|
|
.returns(deferredInfo.promise());
|
|
});
|
|
afterEach(function() {
|
|
createStub.restore();
|
|
getFileInfoStub.restore();
|
|
});
|
|
|
|
it('creates folder with given name and adds it to the list', function() {
|
|
fileList.createDirectory('sub dir');
|
|
|
|
expect(createStub.calledOnce).toEqual(true);
|
|
expect(createStub.getCall(0).args[0]).toEqual('/subdir/sub dir');
|
|
|
|
deferredCreate.resolve(200);
|
|
|
|
expect(getFileInfoStub.calledOnce).toEqual(true);
|
|
expect(getFileInfoStub.getCall(0).args[0]).toEqual('/subdir/sub dir');
|
|
|
|
deferredInfo.resolve(
|
|
200,
|
|
new FileInfo({
|
|
path: '/subdir',
|
|
name: 'sub dir',
|
|
mimetype: 'httpd/unix-directory'
|
|
})
|
|
);
|
|
|
|
var $tr = fileList.findFileEl('sub dir');
|
|
expect($tr.length).toEqual(1);
|
|
expect($tr.attr('data-mime')).toEqual('httpd/unix-directory');
|
|
});
|
|
// TODO: error cases
|
|
// TODO: unique name cases
|
|
});
|
|
/**
|
|
* Test upload mostly by testing the code inside the event handlers
|
|
* that were registered on the magic upload object
|
|
*/
|
|
describe('file upload', function() {
|
|
var $uploader;
|
|
|
|
beforeEach(function() {
|
|
// note: this isn't the real blueimp file uploader from jquery.fileupload
|
|
// but it makes it possible to simulate the event triggering to
|
|
// test the response of the handlers
|
|
$uploader = $('#file_upload_start');
|
|
fileList.setFiles(testFiles);
|
|
});
|
|
|
|
afterEach(function() {
|
|
$uploader = null;
|
|
});
|
|
|
|
describe('dropping external files', function() {
|
|
var uploadData;
|
|
|
|
/**
|
|
* Simulate drop event on the given target
|
|
*
|
|
* @param $target target element to drop on
|
|
* @return event object including the result
|
|
*/
|
|
function dropOn($target, data) {
|
|
var eventData = {
|
|
originalEvent: {
|
|
target: $target
|
|
}
|
|
};
|
|
var ev = new $.Event('fileuploaddrop', eventData);
|
|
// using triggerHandler instead of trigger so we can pass
|
|
// extra data
|
|
$uploader.triggerHandler(ev, data || {});
|
|
return ev;
|
|
}
|
|
|
|
beforeEach(function() {
|
|
// simulate data structure from jquery.upload
|
|
uploadData = {
|
|
files: [{
|
|
relativePath: 'fileToUpload.txt'
|
|
}]
|
|
};
|
|
});
|
|
afterEach(function() {
|
|
uploadData = null;
|
|
});
|
|
it('drop on a tr or crumb outside file list does not trigger upload', function() {
|
|
var $anotherTable = $('<table><tbody><tr><td>outside<div class="crumb">crumb</div></td></tr></table>');
|
|
var ev;
|
|
$('#testArea').append($anotherTable);
|
|
ev = dropOn($anotherTable.find('tr'), uploadData);
|
|
expect(ev.result).toEqual(false);
|
|
|
|
ev = dropOn($anotherTable.find('.crumb'));
|
|
expect(ev.result).toEqual(false);
|
|
});
|
|
it('drop on an element outside file list container does not trigger upload', function() {
|
|
var $anotherEl = $('<div>outside</div>');
|
|
var ev;
|
|
$('#testArea').append($anotherEl);
|
|
ev = dropOn($anotherEl);
|
|
|
|
expect(ev.result).toEqual(false);
|
|
});
|
|
it('drop on an element inside the table triggers upload', function() {
|
|
var ev;
|
|
ev = dropOn(fileList.$fileList.find('th:first'), uploadData);
|
|
|
|
expect(ev.result).not.toEqual(false);
|
|
});
|
|
it('drop on an element on the table container triggers upload', function() {
|
|
var ev;
|
|
ev = dropOn($('#app-content-files'), uploadData);
|
|
|
|
expect(ev.result).not.toEqual(false);
|
|
});
|
|
it('drop on an element inside the table does not trigger upload if no upload permission', function() {
|
|
$('#permissions').val(0);
|
|
var ev;
|
|
ev = dropOn(fileList.$fileList.find('th:first'));
|
|
|
|
expect(ev.result).toEqual(false);
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
});
|
|
it('drop on an folder does not trigger upload if no upload permission on that folder', function() {
|
|
var $tr = fileList.findFileEl('somedir');
|
|
var ev;
|
|
$tr.data('permissions', OC.PERMISSION_READ);
|
|
ev = dropOn($tr);
|
|
|
|
expect(ev.result).toEqual(false);
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
});
|
|
it('drop on a file row inside the table triggers upload to current folder', function() {
|
|
var ev;
|
|
ev = dropOn(fileList.findFileEl('One.txt').find('td:first'), uploadData);
|
|
|
|
expect(ev.result).not.toEqual(false);
|
|
});
|
|
it('drop on a folder row inside the table triggers upload to target folder', function() {
|
|
var ev;
|
|
ev = dropOn(fileList.findFileEl('somedir').find('td:eq(2)'), uploadData);
|
|
|
|
expect(ev.result).not.toEqual(false);
|
|
expect(uploadData.targetDir).toEqual('/subdir/somedir');
|
|
});
|
|
it('drop on a breadcrumb inside the table triggers upload to target folder', function() {
|
|
var ev;
|
|
fileList.changeDirectory('a/b/c/d');
|
|
ev = dropOn(fileList.$el.find('.crumb:eq(2)'), uploadData);
|
|
|
|
expect(ev.result).not.toEqual(false);
|
|
expect(uploadData.targetDir).toEqual('/a/b');
|
|
});
|
|
it('renders upload indicator element for folders only', function() {
|
|
fileList.add({
|
|
name: 'afolder',
|
|
type: 'dir',
|
|
mime: 'httpd/unix-directory'
|
|
});
|
|
fileList.add({
|
|
name: 'afile.txt',
|
|
type: 'file',
|
|
mime: 'text/plain'
|
|
});
|
|
|
|
expect(fileList.findFileEl('afolder').find('.uploadtext').length).toEqual(1);
|
|
expect(fileList.findFileEl('afile.txt').find('.uploadtext').length).toEqual(0);
|
|
});
|
|
});
|
|
});
|
|
describe('Handling errors', function () {
|
|
var deferredList;
|
|
var getFolderContentsStub;
|
|
|
|
beforeEach(function() {
|
|
deferredList = $.Deferred();
|
|
getFolderContentsStub =
|
|
sinon.stub(filesClient, 'getFolderContents');
|
|
getFolderContentsStub.onCall(0).returns(deferredList.promise());
|
|
getFolderContentsStub.onCall(1).returns($.Deferred().promise());
|
|
fileList.reload();
|
|
});
|
|
afterEach(function() {
|
|
getFolderContentsStub.restore();
|
|
fileList = undefined;
|
|
});
|
|
it('redirects to root folder in case of forbidden access', function () {
|
|
deferredList.reject(403);
|
|
|
|
expect(fileList.getCurrentDirectory()).toEqual('/');
|
|
expect(getFolderContentsStub.calledTwice).toEqual(true);
|
|
});
|
|
it('redirects to root folder and shows notification in case of internal server error', function () {
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
deferredList.reject(500);
|
|
|
|
expect(fileList.getCurrentDirectory()).toEqual('/');
|
|
expect(getFolderContentsStub.calledTwice).toEqual(true);
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
});
|
|
it('redirects to root folder and shows notification in case of storage not available', function () {
|
|
expect(notificationStub.notCalled).toEqual(true);
|
|
deferredList.reject(503, 'Storage not available');
|
|
|
|
expect(fileList.getCurrentDirectory()).toEqual('/');
|
|
expect(getFolderContentsStub.calledTwice).toEqual(true);
|
|
expect(notificationStub.calledOnce).toEqual(true);
|
|
});
|
|
});
|
|
describe('showFileBusyState', function() {
|
|
var $tr;
|
|
|
|
beforeEach(function() {
|
|
fileList.setFiles(testFiles);
|
|
$tr = fileList.findFileEl('Two.jpg');
|
|
});
|
|
it('shows spinner on busy rows', function() {
|
|
fileList.showFileBusyState('Two.jpg', true);
|
|
expect($tr.hasClass('busy')).toEqual(true);
|
|
expect(OC.TestUtil.getImageUrl($tr.find('.thumbnail')))
|
|
.toEqual(OC.imagePath('core', 'loading.gif'));
|
|
|
|
fileList.showFileBusyState('Two.jpg', false);
|
|
expect($tr.hasClass('busy')).toEqual(false);
|
|
expect(OC.TestUtil.getImageUrl($tr.find('.thumbnail')))
|
|
.toEqual(OC.imagePath('core', 'filetypes/image.svg'));
|
|
});
|
|
it('accepts multiple input formats', function() {
|
|
_.each([
|
|
'Two.jpg',
|
|
['Two.jpg'],
|
|
$tr,
|
|
[$tr]
|
|
], function(testCase) {
|
|
fileList.showFileBusyState(testCase, true);
|
|
expect($tr.hasClass('busy')).toEqual(true);
|
|
fileList.showFileBusyState(testCase, false);
|
|
expect($tr.hasClass('busy')).toEqual(false);
|
|
});
|
|
});
|
|
});
|
|
describe('elementToFile', function() {
|
|
var $tr;
|
|
|
|
beforeEach(function() {
|
|
fileList.setFiles(testFiles);
|
|
$tr = fileList.findFileEl('One.txt');
|
|
});
|
|
|
|
it('converts data attributes to file info structure', function() {
|
|
var fileInfo = fileList.elementToFile($tr);
|
|
expect(fileInfo.id).toEqual(1);
|
|
expect(fileInfo.name).toEqual('One.txt');
|
|
expect(fileInfo.mtime).toEqual(123456789);
|
|
expect(fileInfo.etag).toEqual('abc');
|
|
expect(fileInfo.permissions).toEqual(OC.PERMISSION_ALL);
|
|
expect(fileInfo.size).toEqual(12);
|
|
expect(fileInfo.mimetype).toEqual('text/plain');
|
|
expect(fileInfo.type).toEqual('file');
|
|
expect(fileInfo.path).not.toBeDefined();
|
|
});
|
|
it('adds path attribute if available', function() {
|
|
$tr.attr('data-path', '/subdir');
|
|
var fileInfo = fileList.elementToFile($tr);
|
|
expect(fileInfo.path).toEqual('/subdir');
|
|
});
|
|
});
|
|
describe('new file menu', function() {
|
|
var newFileMenuStub;
|
|
|
|
beforeEach(function() {
|
|
newFileMenuStub = sinon.stub(OCA.Files.NewFileMenu.prototype, 'showAt');
|
|
});
|
|
afterEach(function() {
|
|
newFileMenuStub.restore();
|
|
})
|
|
it('renders new button when no legacy upload button exists', function() {
|
|
expect(fileList.$el.find('.button.upload').length).toEqual(0);
|
|
expect(fileList.$el.find('.button.new').length).toEqual(1);
|
|
});
|
|
it('does not render new button when no legacy upload button exists (public page)', function() {
|
|
fileList.destroy();
|
|
$('#controls').append('<input type="button" class="button upload" />');
|
|
fileList = new OCA.Files.FileList($('#app-content-files'));
|
|
expect(fileList.$el.find('.button.upload').length).toEqual(1);
|
|
expect(fileList.$el.find('.button.new').length).toEqual(0);
|
|
});
|
|
it('opens the new file menu when clicking on the "New" button', function() {
|
|
var $button = fileList.$el.find('.button.new');
|
|
$button.click();
|
|
expect(newFileMenuStub.calledOnce).toEqual(true);
|
|
});
|
|
it('does not open the new file menu when button is disabled', function() {
|
|
var $button = fileList.$el.find('.button.new');
|
|
$button.addClass('disabled');
|
|
$button.click();
|
|
expect(newFileMenuStub.notCalled).toEqual(true);
|
|
});
|
|
});
|
|
describe('mount type detection', function() {
|
|
function testMountType(dirInfoId, dirInfoMountType, inputMountType, expectedMountType) {
|
|
var $tr;
|
|
fileList.dirInfo.id = dirInfoId;
|
|
fileList.dirInfo.mountType = dirInfoMountType;
|
|
$tr = fileList.add({
|
|
type: 'dir',
|
|
mimetype: 'httpd/unix-directory',
|
|
name: 'test dir',
|
|
mountType: inputMountType
|
|
});
|
|
|
|
expect($tr.attr('data-mounttype')).toEqual(expectedMountType);
|
|
}
|
|
|
|
it('leaves mount type as is if no parent exists', function() {
|
|
testMountType(null, null, 'external', 'external');
|
|
testMountType(null, null, 'shared', 'shared');
|
|
});
|
|
it('detects share root if parent exists', function() {
|
|
testMountType(123, null, 'shared', 'shared-root');
|
|
testMountType(123, 'shared', 'shared', 'shared');
|
|
testMountType(123, 'shared-root', 'shared', 'shared');
|
|
});
|
|
it('detects external storage root if parent exists', function() {
|
|
testMountType(123, null, 'external', 'external-root');
|
|
testMountType(123, 'external', 'external', 'external');
|
|
testMountType(123, 'external-root', 'external', 'external');
|
|
});
|
|
});
|
|
});
|