mirror of
https://github.com/coderaiser/cloudcmd.git
synced 2026-01-23 02:35:49 +00:00
480 lines
11 KiB
JavaScript
480 lines
11 KiB
JavaScript
/* global CloudCmd */
|
|
/* global Util */
|
|
/* global DOM */
|
|
/* global fileop */
|
|
|
|
'use strict';
|
|
|
|
const currify = require('currify/legacy');
|
|
const wraptile = require('wraptile/legacy');
|
|
const {promisify} = require('es6-promisify');
|
|
const exec = require('execon');
|
|
const loadJS = require('load.js').js;
|
|
|
|
const {encode} = require('../../../common/entity');
|
|
|
|
const RESTful = require('../../dom/rest');
|
|
const removeExtension = require('./remove-extension');
|
|
const setListeners = require('./set-listeners');
|
|
const getNextCurrentName = require('./get-next-current-name');
|
|
|
|
const removeQuery = (a) => a.replace(/\?.*/, '');
|
|
|
|
const Name = 'Operation';
|
|
CloudCmd[Name] = exports;
|
|
|
|
const {
|
|
TITLE,
|
|
config,
|
|
} = CloudCmd;
|
|
const {Dialog, Images} = DOM;
|
|
const initOperations = wraptile(_initOperations);
|
|
const authCheck = wraptile(_authCheck);
|
|
|
|
const Operation = {};
|
|
|
|
let Loaded;
|
|
|
|
let {
|
|
cp: copyFn,
|
|
mv: moveFn,
|
|
delete: deleteFn,
|
|
extract: extractFn,
|
|
} = RESTful;
|
|
|
|
let packZipFn = RESTful.pack;
|
|
let packTarFn = RESTful.pack;
|
|
|
|
const Info = DOM.CurrentInfo;
|
|
const showLoad = Images.show.load.bind(null, 'top');
|
|
|
|
const processFiles = currify(_processFiles);
|
|
|
|
const noFilesCheck = () => {
|
|
const {length} = DOM.getActiveFiles();
|
|
const is = Boolean(!length);
|
|
|
|
if (is)
|
|
return Dialog.alert.noFiles(TITLE);
|
|
|
|
return is;
|
|
};
|
|
|
|
module.exports.init = promisify((callback) => {
|
|
showLoad();
|
|
|
|
exec.series([
|
|
DOM.loadSocket,
|
|
(callback) => {
|
|
if (!config('progress') || config('dropbox'))
|
|
return callback();
|
|
|
|
load(initOperations(CloudCmd.prefix, callback));
|
|
},
|
|
(callback) => {
|
|
Loaded = true;
|
|
Images.hide();
|
|
callback();
|
|
},
|
|
], callback);
|
|
});
|
|
|
|
function _authCheck(spawn, ok) {
|
|
const accept = wraptile(ok);
|
|
const alertDialog = wraptile(Dialog.alert);
|
|
|
|
spawn.on('accept', accept(spawn));
|
|
spawn.on('reject', alertDialog (TITLE, 'Wrong credentials!'));
|
|
spawn.emit('auth', config('username'), config('password'));
|
|
}
|
|
|
|
function _initOperations(socketPrefix, fn) {
|
|
const prefix = `${socketPrefix}/fileop`;
|
|
fileop({prefix, socketPrefix}, (e, operator) => {
|
|
fn();
|
|
|
|
operator.on('connect', authCheck(operator, onConnect));
|
|
operator.on('disconnect', onDisconnect);
|
|
});
|
|
}
|
|
|
|
function onConnect(operator) {
|
|
packTarFn = (data, callback) => {
|
|
operator.tar(data.from, data.to, data.names)
|
|
.then(setListeners({noContinue: true}, callback));
|
|
};
|
|
|
|
packZipFn = (data, callback) => {
|
|
operator.zip(data.from, data.to, data.names)
|
|
.then(setListeners({noContinue: true}, callback));
|
|
};
|
|
|
|
deleteFn = (from, files, callback) => {
|
|
from = removeQuery(from);
|
|
operator.remove(from, files)
|
|
.then(setListeners(callback));
|
|
};
|
|
|
|
copyFn = (data, callback) => {
|
|
operator.copy(data.from, data.to, data.names)
|
|
.then(setListeners(callback));
|
|
};
|
|
|
|
moveFn = (data, callback) => {
|
|
operator.move(data.from, data.to, data.names)
|
|
.then(setListeners(callback));
|
|
};
|
|
|
|
extractFn = (data, callback) => {
|
|
operator.extract(data.from, data.to)
|
|
.then(setListeners({noContinue: true}, callback));
|
|
};
|
|
}
|
|
|
|
function onDisconnect() {
|
|
packZipFn = RESTful.pack;
|
|
packTarFn = RESTful.pack;
|
|
deleteFn = RESTful.delete;
|
|
copyFn = RESTful.cp;
|
|
moveFn = RESTful.mv;
|
|
extractFn = RESTful.extract;
|
|
}
|
|
|
|
function getPacker(type) {
|
|
if (type === 'zip')
|
|
return packZipFn;
|
|
|
|
return packTarFn;
|
|
}
|
|
|
|
module.exports.hide = () => {
|
|
CloudCmd.View.hide();
|
|
};
|
|
|
|
module.exports.show = (operation, data) => {
|
|
if (!Loaded)
|
|
return;
|
|
|
|
if (operation === 'copy')
|
|
return Operation.copy(data);
|
|
|
|
if (operation === 'move')
|
|
return Operation.move(data);
|
|
|
|
if (operation === 'delete')
|
|
return Operation.delete();
|
|
|
|
if (operation === 'delete:silent')
|
|
return Operation.deleteSilent();
|
|
|
|
if (operation === 'pack')
|
|
return Operation.pack();
|
|
|
|
if (operation === 'extract')
|
|
return Operation.extract();
|
|
};
|
|
|
|
Operation.copy = processFiles({
|
|
type: 'copy',
|
|
});
|
|
|
|
Operation.move = processFiles({
|
|
type: 'move',
|
|
});
|
|
|
|
Operation.delete = () => {
|
|
promptDelete();
|
|
};
|
|
|
|
Operation.deleteSilent = () => {
|
|
deleteSilent();
|
|
};
|
|
|
|
Operation.pack = () => {
|
|
const isZip = config('packer') === 'zip';
|
|
twopack('pack', isZip ? 'zip' : 'tar');
|
|
};
|
|
|
|
Operation.extract = () => {
|
|
twopack('extract');
|
|
};
|
|
|
|
/**
|
|
* prompt and delete current file or selected files
|
|
*
|
|
* @currentFile
|
|
*/
|
|
function promptDelete() {
|
|
if (noFilesCheck())
|
|
return;
|
|
|
|
const msgAsk = 'Do you really want to delete the ';
|
|
const msgSel = 'selected ';
|
|
|
|
const files = DOM.getActiveFiles();
|
|
const names = DOM.getFilenames(files);
|
|
const n = names.length;
|
|
|
|
let msg;
|
|
if (n) {
|
|
let name = '';
|
|
|
|
for (let i = 0; i < 5 && i < n; i++)
|
|
name += '\n' + names[i];
|
|
|
|
if (n >= 5)
|
|
name += '\n...';
|
|
|
|
msg = msgAsk + msgSel + n + ' files/directories?\n' + encode(name);
|
|
} else {
|
|
const current = DOM.getCurrentFile();
|
|
const isDir = DOM.isCurrentIsDir(current);
|
|
const getType = (isDir) => {
|
|
return isDir ? 'directory' : 'file';
|
|
};
|
|
|
|
const type = getType(isDir) + ' ';
|
|
|
|
const name = DOM.getCurrentName(current);
|
|
msg = msgAsk + msgSel + type + name + '?';
|
|
}
|
|
|
|
const cancel = false;
|
|
|
|
Dialog.confirm(TITLE, msg, {cancel}).then(() => {
|
|
deleteSilent(files);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* delete current or selected files
|
|
*
|
|
* @files
|
|
*/
|
|
function deleteSilent(files = DOM.getActiveFiles()) {
|
|
const query = '?files';
|
|
const path = Info.dirPath;
|
|
|
|
if (noFilesCheck())
|
|
return;
|
|
|
|
showLoad();
|
|
|
|
const removedNames = DOM.getFilenames(files);
|
|
const names = DOM.CurrentInfo.files.map(DOM.getCurrentName);
|
|
const currentName = DOM.getCurrentName();
|
|
const nextCurrentName = getNextCurrentName(currentName, names, removedNames);
|
|
|
|
deleteFn(path + query, removedNames, () => {
|
|
CloudCmd.refresh(() => {
|
|
const names = Info.files.map(DOM.getCurrentName);
|
|
const isCurrent = names.includes(currentName);
|
|
|
|
const name = isCurrent ? currentName : nextCurrentName;
|
|
|
|
DOM.setCurrentByName(name);
|
|
});
|
|
});
|
|
}
|
|
|
|
/*
|
|
* process files (copy or move)
|
|
* @param data
|
|
* @param operation
|
|
*/
|
|
function _processFiles(options, data) {
|
|
let selFiles;
|
|
let files;
|
|
let panel;
|
|
let shouldAsk;
|
|
let ok;
|
|
|
|
let from = '';
|
|
let to = '';
|
|
|
|
let names = [];
|
|
|
|
/* eslint no-multi-spaces: 0 */
|
|
|
|
if (data) {
|
|
from = data.from;
|
|
to = data.to;
|
|
names = data.names;
|
|
panel = Info.panel;
|
|
} else {
|
|
from = Info.dirPath;
|
|
to = DOM.getNotCurrentDirPath();
|
|
selFiles = DOM.getSelectedFiles();
|
|
names = DOM.getFilenames(selFiles);
|
|
data = {};
|
|
shouldAsk = true;
|
|
panel = Info.panelPassive;
|
|
}
|
|
|
|
if (!names.length)
|
|
names.push(DOM.getCurrentName());
|
|
|
|
const [name] = names;
|
|
|
|
const sameName = DOM.getCurrentByName(name, panel);
|
|
|
|
if (!data && noFilesCheck())
|
|
return;
|
|
|
|
const {type} = options;
|
|
|
|
const isCopy = type === 'copy';
|
|
const option = isCopy ? 'confirmCopy' : 'confirmMove';
|
|
const title = isCopy ? 'Copy' : 'Rename/Move';
|
|
const operation = isCopy ? copyFn : moveFn;
|
|
|
|
if (shouldAsk && config(option))
|
|
return message(title, to, names.map(encode))
|
|
.then(ask);
|
|
|
|
ask(to);
|
|
|
|
function ask(to) {
|
|
ok = from !== to && to;
|
|
|
|
if (ok && !shouldAsk || !sameName)
|
|
return go();
|
|
|
|
const str = `"${ name }" already exist. Overwrite?`;
|
|
const cancel = false;
|
|
|
|
Dialog.confirm(TITLE, str, {cancel}).then(go);
|
|
|
|
function go() {
|
|
showLoad();
|
|
|
|
files = {
|
|
from,
|
|
to,
|
|
names,
|
|
};
|
|
|
|
operation(files, () => {
|
|
DOM.Storage.remove(from, () => {
|
|
const {
|
|
panel,
|
|
panelPassive,
|
|
} = Info;
|
|
|
|
const setCurrent = () => {
|
|
const currentName = name || data.names[0];
|
|
DOM.setCurrentByName(currentName);
|
|
};
|
|
|
|
if (!Info.isOnePanel)
|
|
CloudCmd.refresh({
|
|
panel: panelPassive,
|
|
noCurrent: true,
|
|
});
|
|
|
|
CloudCmd.refresh({panel}, setCurrent);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
function checkEmpty(name, operation) {
|
|
if (!operation)
|
|
throw Error(name + ' could not be empty!');
|
|
}
|
|
|
|
function twopack(operation, type) {
|
|
let op;
|
|
let fileFrom;
|
|
let currentName = Info.name;
|
|
|
|
const {Images} = DOM;
|
|
|
|
const {
|
|
path,
|
|
dirPath,
|
|
} = Info;
|
|
|
|
const activeFiles = DOM.getActiveFiles();
|
|
const names = DOM.getFilenames(activeFiles);
|
|
|
|
checkEmpty('operation', operation);
|
|
|
|
if (!names.length)
|
|
return Dialog.alert.noFiles(TITLE);
|
|
|
|
switch(operation) {
|
|
case 'extract':
|
|
op = extractFn;
|
|
|
|
fileFrom = {
|
|
from: path,
|
|
to: dirPath,
|
|
};
|
|
|
|
currentName = removeExtension(currentName);
|
|
|
|
break;
|
|
|
|
case 'pack':
|
|
op = getPacker(type);
|
|
|
|
if (names.length > 1)
|
|
currentName = Info.dir;
|
|
|
|
currentName += DOM.getPackerExt(type);
|
|
|
|
fileFrom = {
|
|
from: dirPath,
|
|
to: dirPath + currentName,
|
|
names,
|
|
};
|
|
break;
|
|
}
|
|
|
|
Images.show.load('top');
|
|
|
|
op(fileFrom, (error) => {
|
|
!error && CloudCmd.refresh({
|
|
currentName,
|
|
});
|
|
});
|
|
}
|
|
|
|
function message(msg, to, names) {
|
|
const n = names.length;
|
|
const [name] = names;
|
|
|
|
msg += ' ';
|
|
|
|
if (names.length > 1)
|
|
msg += n + ' file(s)';
|
|
else
|
|
msg += '"' + name + '"';
|
|
|
|
msg += ' to';
|
|
|
|
const cancel = false;
|
|
|
|
return Dialog.prompt(TITLE, msg, to, {cancel});
|
|
}
|
|
|
|
function load(callback) {
|
|
const {prefix} = CloudCmd;
|
|
const file = `${prefix}/fileop/fileop.js`;
|
|
|
|
loadJS(file, (error) => {
|
|
if (error) {
|
|
Dialog.alert(TITLE, error.message);
|
|
return exec(callback);
|
|
}
|
|
|
|
Loaded = true;
|
|
Util.timeEnd(Name + ' load');
|
|
exec(callback);
|
|
});
|
|
|
|
Util.time(Name + ' load');
|
|
}
|
|
|