mirror of
				https://github.com/mickael-kerjean/filestash.git
				synced 2025-11-04 05:27:04 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			443 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			443 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
import React from 'react';
 | 
						|
 | 
						|
import { Files } from '../model/';
 | 
						|
import { notify, alert, currentShare } from '../helpers/';
 | 
						|
import Path from 'path';
 | 
						|
import Worker from "../worker/search.worker.js";
 | 
						|
import { Observable } from "rxjs/Observable";
 | 
						|
 | 
						|
export const sort = function(files, type){
 | 
						|
    if(type === 'name'){
 | 
						|
        return sortByName(files);
 | 
						|
    }else if(type === 'date'){
 | 
						|
        return sortByDate(files);
 | 
						|
    }else{
 | 
						|
        return sortByType(files);
 | 
						|
    }
 | 
						|
    function sortByType(files){
 | 
						|
        return files.sort((fileA, fileB) => {
 | 
						|
            if(fileA.icon === 'loading' && fileB.icon !== 'loading'){return +1;}
 | 
						|
            else if(fileA.icon !== 'loading' && fileB.icon === 'loading'){return -1;}
 | 
						|
            else{
 | 
						|
                if(['directory', 'link'].indexOf(fileA.type) === -1 && ['directory', 'link'].indexOf(fileB.type) !== -1){
 | 
						|
                    return +1;
 | 
						|
                }else if(['directory', 'link'].indexOf(fileA.type) !== -1 && ['directory', 'link'].indexOf(fileB.type) === -1){
 | 
						|
                    return -1;
 | 
						|
                }else{
 | 
						|
                    if(fileA.name[0] === "." && fileB.name[0] !== ".") return +1;
 | 
						|
                    else if(fileA.name[0] !== "." && fileB.name[0] === ".") return -1;
 | 
						|
                    else{
 | 
						|
                        const aExt = Path.extname(fileA.name.toLowerCase()),
 | 
						|
                              bExt = Path.extname(fileB.name.toLowerCase());
 | 
						|
                        if(fileA.name.toLowerCase() === fileB.name.toLowerCase()){
 | 
						|
                            return fileA.name > fileB.name ? +1 : -1;
 | 
						|
                        }else{
 | 
						|
                            if(aExt !== bExt) return aExt > bExt ? +1 : -1;
 | 
						|
                            else return fileA.name.toLowerCase() > fileB.name.toLowerCase() ? +1 : -1;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
    function sortByName(files){
 | 
						|
        return files.sort((fileA, fileB) => {
 | 
						|
            if(fileA.icon === 'loading' && fileB.icon !== 'loading'){return +1;}
 | 
						|
            else if(fileA.icon !== 'loading' && fileB.icon === 'loading'){return -1;}
 | 
						|
            else{
 | 
						|
                if(fileA.name[0] === "." && fileB.name[0] !== ".") return +1;
 | 
						|
                else if(fileA.name[0] !== "." && fileB.name[0] === ".") return -1;
 | 
						|
                else{
 | 
						|
                    if(fileA.name.toLowerCase() === fileB.name.toLowerCase()){
 | 
						|
                        return fileA.name > fileB.name ? +1 : -1;
 | 
						|
                    }
 | 
						|
                    return fileA.name.toLowerCase() > fileB.name.toLowerCase() ? +1 : -1;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
    function sortByDate(files){
 | 
						|
        return files.sort((fileA, fileB) => {
 | 
						|
            if(fileA.icon === 'loading' && fileB.icon !== 'loading'){return +1;}
 | 
						|
            else if(fileA.icon !== 'loading' && fileB.icon === 'loading'){return -1;}
 | 
						|
            else{
 | 
						|
                if(fileB.time === fileA.time){
 | 
						|
                    return fileA.name > fileB.name ? +1 : -1;
 | 
						|
                }
 | 
						|
                return fileB.time - fileA.time;
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
export const onCreate = function(path, type, file){
 | 
						|
    if(type === 'file'){
 | 
						|
        return Files.touch(path, file)
 | 
						|
            .then(() => {
 | 
						|
                notify.send('A file named "'+Path.basename(path)+'" was created', 'success');
 | 
						|
                return Promise.resolve();
 | 
						|
            })
 | 
						|
            .catch((err) => {
 | 
						|
                notify.send(err, 'error');
 | 
						|
                return Promise.reject(err);
 | 
						|
            });
 | 
						|
    }else if(type === 'directory'){
 | 
						|
        return Files.mkdir(path)
 | 
						|
            .then(() => notify.send('A folder named "'+Path.basename(path)+'" was created', 'success'))
 | 
						|
            .catch((err) => notify.send(err, 'error'));
 | 
						|
    }else{
 | 
						|
        return Promise.reject({message: 'internal error: can\'t create a '+type.toString(), code: 'UNKNOWN_TYPE'});
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
export const onRename = function(from, to, type){
 | 
						|
    return Files.mv(from, to, type)
 | 
						|
        .then(() => notify.send('The file "'+Path.basename(from)+'" was renamed', 'success'))
 | 
						|
        .catch((err) => notify.send(err, 'error'));
 | 
						|
};
 | 
						|
 | 
						|
export const onDelete = function(path, type){
 | 
						|
    return Files.rm(path, type)
 | 
						|
        .then(() => notify.send('The file "'+Path.basename(path)+'" was deleted', 'success'))
 | 
						|
        .catch((err) => notify.send(err, 'error'));
 | 
						|
};
 | 
						|
 | 
						|
export const onMultiDelete = function(arrOfPath){
 | 
						|
    return Promise.all(arrOfPath.map((p) => Files.rm(p)))
 | 
						|
        .then(() => notify.send('All done!', 'success'))
 | 
						|
        .catch((err) => notify.send(err, 'error'));
 | 
						|
}
 | 
						|
 | 
						|
export const onMultiRename = function(arrOfPath){
 | 
						|
    return Promise.all(arrOfPath.map((p) => Files.mv(p[0], p[1])))
 | 
						|
        .then(() => notify.send('All done!', 'success'))
 | 
						|
        .catch((err) => notify.send(err, 'error'));
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * The upload method has a few strategies:
 | 
						|
 * 1. user is coming from drag and drop + browser provides support to read entire folders
 | 
						|
 * 2. user is coming from drag and drop + browser DOES NOT provides support to read entire folders
 | 
						|
 * 3. user is coming from a upload form button as he doesn't have drag and drop with files
 | 
						|
 */
 | 
						|
export const onUpload = function(path, e){
 | 
						|
    const MAX_POOL_SIZE = 15;
 | 
						|
    let PRIOR_STATUS = {};
 | 
						|
 | 
						|
    let extractFiles = null;
 | 
						|
    if(e.dataTransfer === undefined){ // case 3
 | 
						|
        extractFiles = extract_upload_crappy_hack_but_official_way(e.target);
 | 
						|
    } else {
 | 
						|
        if(e.dataTransfer.types && e.dataTransfer.types.length >= 0){
 | 
						|
            if(e.dataTransfer.types[0] === "text/uri-list"){
 | 
						|
                return
 | 
						|
            }
 | 
						|
        }
 | 
						|
        extractFiles = extract_upload_directory_the_way_that_works_but_non_official(e.dataTransfer.items || [], []) // case 1
 | 
						|
            .then((files) => {
 | 
						|
                if(files.length === 0){ // case 2
 | 
						|
                    return extract_upload_crappy_hack_but_official_way(e.dataTransfer);
 | 
						|
                }
 | 
						|
                return Promise.resolve(files);
 | 
						|
            })
 | 
						|
    }
 | 
						|
 | 
						|
    extractFiles.then((files) => {
 | 
						|
            var failed = [],
 | 
						|
                currents = [];
 | 
						|
 | 
						|
            const processes = files.map((file) => {
 | 
						|
                let original_path = file.path;
 | 
						|
                file.path = Path.join(path, file.path);
 | 
						|
                if(file.type === 'file'){
 | 
						|
                    if(files.length < 150) Files.touch(file.path, file.file, 'prepare_only');
 | 
						|
                    return {
 | 
						|
                        path: original_path,
 | 
						|
                        parent: file._prior || null,
 | 
						|
                        fn: Files.touch.bind(Files, file.path, file.file, 'execute_only')
 | 
						|
                    };
 | 
						|
                }else{
 | 
						|
                    Files.mkdir(file.path, 'prepare_only');
 | 
						|
                    return {
 | 
						|
                        id: file._id || null,
 | 
						|
                        path: original_path,
 | 
						|
                        parent: file._prior || null,
 | 
						|
                        fn: Files.mkdir.bind(Files, file.path, 'execute_only')
 | 
						|
                    };
 | 
						|
                }
 | 
						|
            });
 | 
						|
            class Stats extends React.Component {
 | 
						|
                constructor(props){
 | 
						|
                    super(props);
 | 
						|
                    this.state = {timeout: 1};
 | 
						|
                }
 | 
						|
 | 
						|
                componentDidMount(){
 | 
						|
                    if(typeof this.state.timeout === "number"){
 | 
						|
                        this.setState({
 | 
						|
                            timeout: window.setTimeout(() => {
 | 
						|
                                this.componentDidMount();
 | 
						|
                            }, Math.random()*1000+200)
 | 
						|
                        });
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                componentWillUnmount(){
 | 
						|
                    window.clearTimeout(this.state.timeout);
 | 
						|
                }
 | 
						|
 | 
						|
                emphasis(path){
 | 
						|
                    notify.send(path.split("/").join(" / "), "info");
 | 
						|
                }
 | 
						|
 | 
						|
                render() {
 | 
						|
                    const percent = Math.floor(100 * (files.length - processes.length - currents.length) / files.length);
 | 
						|
                    return (
 | 
						|
                        <div className="component_stats">
 | 
						|
                          <h2>
 | 
						|
                            UPLOADING <span className="percent">({percent}%)</span>
 | 
						|
                            <div>
 | 
						|
                              <span className="completed">{files.length - processes.length - currents.length}</span>
 | 
						|
                              <span className="grandTotal">{files.length}</span>
 | 
						|
                            </div>
 | 
						|
                          </h2>
 | 
						|
                          <div className="stats_content">
 | 
						|
                          {
 | 
						|
                              currents.slice(0, 1000).map((process, i) => {
 | 
						|
                                  return (
 | 
						|
                                      <div onClick={() => this.emphasis(process.path)} className="current_color" key={i}>{process.path.replace(/\//, '')}</div>
 | 
						|
                                  );
 | 
						|
                              })
 | 
						|
                          }
 | 
						|
                          {
 | 
						|
                              processes.slice(0, 1000).map((process, i) => {
 | 
						|
                                  return (
 | 
						|
                                      <div onClick={() => this.emphasis(process.path)} className="todo_color" key={i}>{process.path.replace(/\//, '')}</div>
 | 
						|
                                  );
 | 
						|
                              })
 | 
						|
                          }
 | 
						|
                          {
 | 
						|
                              failed.slice(0, 500).map((process, i) => {
 | 
						|
                                  return (
 | 
						|
                                      <div onClick={() => this.emphasis(process.path)} className="error_color" key={i}>{process.path}</div>
 | 
						|
                                  );
 | 
						|
                              })
 | 
						|
                          }
 | 
						|
                          </div>
 | 
						|
                        </div>
 | 
						|
                    );
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            function runner(id){
 | 
						|
                let current_process = null;
 | 
						|
                if(processes.length === 0) return Promise.resolve();
 | 
						|
 | 
						|
                var i;
 | 
						|
                for(i=0; i<processes.length; i++){
 | 
						|
                    if(
 | 
						|
                        // init: getting started with creation of files/folders
 | 
						|
                        processes[i].parent === null ||
 | 
						|
                        // running: make sure we've created the parent folder
 | 
						|
                        PRIOR_STATUS[processes[i].parent] === true
 | 
						|
                    ){
 | 
						|
                        current_process = processes[i];
 | 
						|
                        processes.splice(i, 1);
 | 
						|
                        currents.push(current_process);
 | 
						|
                        break;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if(current_process){
 | 
						|
                    return current_process.fn(id)
 | 
						|
                        .then(() => {
 | 
						|
                            if(current_process.id) PRIOR_STATUS[current_process.id] = true;
 | 
						|
                            currents = currents.filter((c) => c.path != current_process.path);
 | 
						|
                            return runner(id);
 | 
						|
                        })
 | 
						|
                        .catch((err) => {
 | 
						|
                            failed.push(current_process);
 | 
						|
                            currents = currents.filter((c) => c.path != current_process.path);
 | 
						|
                            notify.send(err, 'error');
 | 
						|
                            return runner(id);
 | 
						|
                        });
 | 
						|
                }else{
 | 
						|
                    return waitABit()
 | 
						|
                        .then(() => runner(id));
 | 
						|
 | 
						|
                    function waitABit(){
 | 
						|
                        return new Promise((done) => {
 | 
						|
                            window.setTimeout(() => {
 | 
						|
                                requestAnimationFrame(() => {
 | 
						|
                                    done();
 | 
						|
                                });
 | 
						|
                            }, 250);
 | 
						|
                        });
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            if(files.length >= 5){
 | 
						|
                alert.now(<Stats/>, () => {});
 | 
						|
            }
 | 
						|
            Promise.all(Array.apply(null, Array(MAX_POOL_SIZE)).map((process,index) => {
 | 
						|
                return runner();
 | 
						|
            })).then(() => {
 | 
						|
                // remove the popup
 | 
						|
                if(failed.length === 0){
 | 
						|
                    var e = new Event("keydown");
 | 
						|
                    e.keyCode = 27;
 | 
						|
                    window.dispatchEvent(e);
 | 
						|
                }
 | 
						|
                currents = [];
 | 
						|
                // display message
 | 
						|
                window.setTimeout(() => {
 | 
						|
                    notify.send('Upload completed', 'success');
 | 
						|
                }, 300);
 | 
						|
            }).catch((err) => {
 | 
						|
                currents = [];
 | 
						|
                notify.send(err, 'error');
 | 
						|
            });
 | 
						|
        });
 | 
						|
 | 
						|
 | 
						|
 | 
						|
    // adapted from: https://stackoverflow.com/questions/105034/create-guid-uuid-in-javascript
 | 
						|
    function _rand_id(){
 | 
						|
        function s4() {
 | 
						|
            return Math.floor((1 + Math.random()) * 0x10000)
 | 
						|
                .toString(16)
 | 
						|
                .substring(1);
 | 
						|
        }
 | 
						|
        return s4() + s4() + s4() + s4();
 | 
						|
    }
 | 
						|
 | 
						|
    function extract_upload_directory_the_way_that_works_but_non_official(items, files = []){
 | 
						|
        const traverseDirectory = (item, _files, parent_id) => {
 | 
						|
            let file = {
 | 
						|
                path: item.fullPath
 | 
						|
            };
 | 
						|
            if(item.isFile){
 | 
						|
                return new Promise((done, err) => {
 | 
						|
                    file.type = "file";
 | 
						|
                    item.file((_file, _err) => {
 | 
						|
                        if(!_err){
 | 
						|
                            file.file = _file;
 | 
						|
                            if(parent_id) file._prior = parent_id;
 | 
						|
                            _files.push(file);
 | 
						|
                        }
 | 
						|
                        done(_files);
 | 
						|
                    });
 | 
						|
                });
 | 
						|
            }else if(item.isDirectory){
 | 
						|
                file.type = "directory";
 | 
						|
                file.path += "/";
 | 
						|
                file._id = _rand_id();
 | 
						|
                if(parent_id) file._prior = parent_id;
 | 
						|
                _files.push(file);
 | 
						|
 | 
						|
                let reader = item.createReader();
 | 
						|
                const filereader = function(r){
 | 
						|
                    return new Promise((done) => {
 | 
						|
                        r.readEntries(function(entries){
 | 
						|
                            Promise.all(entries.map((entry) => {
 | 
						|
                                return traverseDirectory(entry, _files, file._id);
 | 
						|
                            })).then((e) => {
 | 
						|
                                if(entries.length > 0){
 | 
						|
                                    return filereader(r).then(done);
 | 
						|
                                }
 | 
						|
                                return done(e);
 | 
						|
                            })
 | 
						|
                        });
 | 
						|
                    });
 | 
						|
                }
 | 
						|
                return filereader(reader).then(() => {
 | 
						|
                    return Promise.resolve(_files)
 | 
						|
                });
 | 
						|
            }else{
 | 
						|
                return Promise.resolve();
 | 
						|
            }
 | 
						|
        };
 | 
						|
        return Promise.all(
 | 
						|
            Array.prototype.slice.call(items).map((item) => {
 | 
						|
                if(typeof item.webkitGetAsEntry === 'function'){
 | 
						|
                    return traverseDirectory(item.webkitGetAsEntry(), files.slice(0));
 | 
						|
                }
 | 
						|
            }).filter((e) => e)
 | 
						|
        ).then((res) => Promise.resolve([].concat.apply([], res)));
 | 
						|
    }
 | 
						|
 | 
						|
    function extract_upload_crappy_hack_but_official_way(data){
 | 
						|
        const _files = data.files;
 | 
						|
        return Promise.all(
 | 
						|
            Array.prototype.slice.call(_files).map((_file) => {
 | 
						|
                return detectType(_file)
 | 
						|
                    .then(transform);
 | 
						|
                function detectType(_f){
 | 
						|
                    // the 4096 is an heuristic I've observed and taken from: https://stackoverflow.com/questions/25016442/how-to-distinguish-if-a-file-or-folder-is-being-dragged-prior-to-it-being-droppe
 | 
						|
                    // however the proposed answer is just wrong as it doesn't consider folder with name such as: test.png
 | 
						|
                    // and as Stackoverflow favor consanguinity with their point system, I couldn't rectify the proposed answer.
 | 
						|
                    // The following code is actually working as expected
 | 
						|
                    if(_file.size % 4096 !== 0){
 | 
						|
                        return Promise.resolve('file');
 | 
						|
                    }
 | 
						|
                    return new Promise((done, err) => {
 | 
						|
                        let reader = new window.FileReader();
 | 
						|
                        reader.onload = function() {
 | 
						|
                            done('file');
 | 
						|
                        };
 | 
						|
                        reader.onerror = function() {
 | 
						|
                            done('directory');
 | 
						|
                        };
 | 
						|
                        reader.readAsText(_f);
 | 
						|
                    });
 | 
						|
                }
 | 
						|
 | 
						|
                function transform(_type){
 | 
						|
                    let file = {
 | 
						|
                        type: _type,
 | 
						|
                        path: _file.name
 | 
						|
                    };
 | 
						|
                    if(file.type === 'file'){
 | 
						|
                        file.file = _file;
 | 
						|
                    }else{
 | 
						|
                        file.path += "/";
 | 
						|
                    }
 | 
						|
                    return Promise.resolve(file);
 | 
						|
                }
 | 
						|
            })
 | 
						|
        );
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
const worker = new Worker();
 | 
						|
export const onSearch = (keyword, path = "/") => {
 | 
						|
    if(navigator.onLine == false){
 | 
						|
        notify.send("Result aren't complete because you're not online", "info");
 | 
						|
        worker.postMessage({
 | 
						|
            action: "search::find",
 | 
						|
            path: path,
 | 
						|
            share: currentShare(),
 | 
						|
            keyword: keyword
 | 
						|
        });
 | 
						|
        return new Observable((obs) => {
 | 
						|
            worker.onmessage = (m) => {
 | 
						|
                if(m.data.type === "search::found"){
 | 
						|
                    obs.next(m.data && m.data.files || []);
 | 
						|
                }
 | 
						|
            };
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    return new Observable((obs) => {
 | 
						|
        Files.search(keyword, path).then((f) => obs.next(f))
 | 
						|
    });
 | 
						|
};
 | 
						|
 | 
						|
export const createLink = (type, path) => {
 | 
						|
    return type === "file" ? "/view" + path : "/files" + path;
 | 
						|
}
 |