mirror of
				https://github.com/mickael-kerjean/filestash.git
				synced 2025-11-04 13:35:46 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			171 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
import React from 'react';
 | 
						|
import PropTypes from 'prop-types';
 | 
						|
import ReactCSSTransitionGroup from 'react-addons-css-transition-group';
 | 
						|
 | 
						|
import { Card, NgIf, Icon, EventEmitter, Dropdown, DropdownButton, DropdownList, DropdownItem, Container } from '../../components/';
 | 
						|
import { pathBuilder, debounce, prompt } from '../../helpers/';
 | 
						|
import "./submenu.scss";
 | 
						|
 | 
						|
@EventEmitter
 | 
						|
export class Submenu extends React.Component {
 | 
						|
    constructor(props){
 | 
						|
        super(props);
 | 
						|
        this.state = {
 | 
						|
            search_input_visible: false,
 | 
						|
            search_keyword: ""
 | 
						|
        };
 | 
						|
 | 
						|
        this.onSearchChange_Backpressure = debounce(this.onSearchChange, 400);
 | 
						|
        this._onEscapeKeyPress = (e) => {
 | 
						|
            if(e.keyCode === 27){ // escape key
 | 
						|
                this.setState({
 | 
						|
                    search_keyword: "",
 | 
						|
                    search_input_visible: false
 | 
						|
                });
 | 
						|
                this.refs.$input.blur();
 | 
						|
                this.props.onSearch(null);
 | 
						|
            }else if(e.ctrlKey && e.keyCode === 70){ // 'Ctrl F' shortcut to search
 | 
						|
                e.preventDefault();
 | 
						|
                this.setState({
 | 
						|
                    search_input_visible: true
 | 
						|
                });
 | 
						|
                this.refs.$input.focus();
 | 
						|
            }else if(e.altKey && (e.keyCode === 49 || e.keyCode === 50)){ // 'alt 1' 'alt 2' shortcut
 | 
						|
                e.preventDefault();
 | 
						|
                this.onViewChange();
 | 
						|
            }
 | 
						|
        };
 | 
						|
    }
 | 
						|
 | 
						|
    componentDidMount(){
 | 
						|
        window.addEventListener('keydown', this._onEscapeKeyPress);
 | 
						|
        if(this.state.search_input_visible === true){
 | 
						|
            this.onSearchChange(this.state.search_keyword);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    componentWillUnmount(){
 | 
						|
        window.removeEventListener('keydown', this._onEscapeKeyPress);
 | 
						|
    }
 | 
						|
 | 
						|
    onNew(type){
 | 
						|
        this.props.emit("new::"+type);
 | 
						|
    }
 | 
						|
    onDelete(arrayOfPaths){
 | 
						|
        prompt.now(
 | 
						|
            "Confirm by typing \"remove\"",
 | 
						|
            (answer) => {
 | 
						|
                if(answer !== "remove"){
 | 
						|
                    return Promise.resolve();
 | 
						|
                }
 | 
						|
                this.props.emit("file.delete.multiple", arrayOfPaths);
 | 
						|
                return Promise.resolve();
 | 
						|
            },
 | 
						|
            () => { /* click on cancel */ }
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    onViewChange(){
 | 
						|
        requestAnimationFrame(() => this.props.onViewUpdate());
 | 
						|
    }
 | 
						|
 | 
						|
    onSortChange(e){
 | 
						|
        this.props.onSortUpdate(e);
 | 
						|
    }
 | 
						|
 | 
						|
    onSearchChange(search, e){
 | 
						|
        this.props.onSearch(search.trim());
 | 
						|
    }
 | 
						|
 | 
						|
    onSearchToggle(){
 | 
						|
        if(new Date () - this.search_last_toggle < 200){
 | 
						|
            // avoid bluring event cancelling out the toogle
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        this.refs.$input.focus();
 | 
						|
        this.setState({search_input_visible: !this.state.search_input_visible}, () => {
 | 
						|
            if(this.state.search_input_visible == false){
 | 
						|
                this.props.onSearch(null);
 | 
						|
                this.setState({search_keyword: ""});
 | 
						|
            }
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    closeIfEmpty(){
 | 
						|
        if(this.state.search_keyword.trim().length > 0) return;
 | 
						|
        this.search_last_toggle = new Date();
 | 
						|
        this.setState({
 | 
						|
            search_input_visible: false,
 | 
						|
            search_keyword: ""
 | 
						|
        });
 | 
						|
        this.props.onSearch(null);
 | 
						|
    }
 | 
						|
 | 
						|
    onSearchKeypress(s, backpressure = true, e){
 | 
						|
        if(backpressure){
 | 
						|
            this.onSearchChange_Backpressure(s);
 | 
						|
        }else{
 | 
						|
            this.onSearchChange(s);
 | 
						|
        }
 | 
						|
        this.setState({search_keyword: s});
 | 
						|
 | 
						|
        if(e && e.preventDefault){
 | 
						|
            e.preventDefault();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    render(){
 | 
						|
        return (
 | 
						|
            <div className="component_submenu">
 | 
						|
              <Container>
 | 
						|
                <div className={"menubar no-select "+(this.state.search_input_visible ? "search_focus" : "")}>
 | 
						|
                  <NgIf cond={this.props.accessRight.can_create_file !== false && this.props.selected.length === 0} onClick={this.onNew.bind(this, 'file')} type="inline">
 | 
						|
                    New File
 | 
						|
                  </NgIf>
 | 
						|
                  <NgIf cond={this.props.accessRight.can_create_directory !== false && this.props.selected.length === 0} onClick={this.onNew.bind(this, 'directory')} type="inline">
 | 
						|
                    New Directory
 | 
						|
                  </NgIf>
 | 
						|
                  <NgIf cond={this.props.selected.length > 0} type="inline" onMouseDown={this.onDelete.bind(this, this.props.selected)}>
 | 
						|
                    <ReactCSSTransitionGroup transitionName="submenuwithSelection" transitionLeave={false} transitionEnter={false} transitionAppear={true} transitionAppearTimeout={10000}>
 | 
						|
                      <span>Remove</span>
 | 
						|
                    </ReactCSSTransitionGroup>
 | 
						|
                  </NgIf>
 | 
						|
 | 
						|
                  <Dropdown className="view sort" onChange={this.onSortChange.bind(this)}>
 | 
						|
                    <DropdownButton>
 | 
						|
                      <Icon name="sort"/>
 | 
						|
                    </DropdownButton>
 | 
						|
                    <DropdownList>
 | 
						|
                      <DropdownItem name="type" icon={this.props.sort === "type" ? "check" : null}> Sort By Type </DropdownItem>
 | 
						|
                      <DropdownItem name="date" icon={this.props.sort === "date" ? "check" : null}> Sort By Date </DropdownItem>
 | 
						|
                      <DropdownItem name="name" icon={this.props.sort === "name" ? "check" : null}> Sort By Name </DropdownItem>
 | 
						|
                    </DropdownList>
 | 
						|
                  </Dropdown>
 | 
						|
                  <div className="view list-grid" onClick={this.onViewChange.bind(this)}><Icon name={this.props.view === "grid" ? "list" : "grid"}/></div>
 | 
						|
                  <NgIf cond={window.CONFIG.enable_search === true} className="view">
 | 
						|
                    <form onSubmit={(e) => this.onSearchKeypress(this.state.search_keyword, false, e)}>
 | 
						|
                      <label className="view search" onClick={this.onSearchToggle.bind(this, null)}>
 | 
						|
                        <NgIf cond={this.state.search_input_visible !== true}>
 | 
						|
                          <Icon name="search_dark"/>
 | 
						|
                        </NgIf>
 | 
						|
                        <NgIf cond={this.state.search_input_visible === true}>
 | 
						|
                          <Icon name="close_dark"/>
 | 
						|
                        </NgIf>
 | 
						|
                      </label>
 | 
						|
                      <NgIf cond={this.state.search_input_visible !== null} type="inline">
 | 
						|
                        <input ref="$input" onBlur={this.closeIfEmpty.bind(this, false)} style={{"width": this.state.search_input_visible ? "180px" : "0px"}} value={this.state.search_keyword} onChange={(e) => this.onSearchKeypress(e.target.value, true)} type="text" id="search" placeholder="search" name="search" autoComplete="off" />
 | 
						|
                      </NgIf>
 | 
						|
                    </form>
 | 
						|
                  </NgIf>
 | 
						|
                </div>
 | 
						|
              </Container>
 | 
						|
            </div>
 | 
						|
        );
 | 
						|
    };
 | 
						|
}
 | 
						|
 | 
						|
Submenu.propTypes = {
 | 
						|
    accessRight: PropTypes.object,
 | 
						|
    onSortUpdate: PropTypes.func.isRequired,
 | 
						|
    sort: PropTypes.string.isRequired
 | 
						|
};
 |