From fc4f6f63989183f766f59e262194fb4c398c0d28 Mon Sep 17 00:00:00 2001 From: nathangray Date: Mon, 7 Jan 2019 15:04:50 -0700 Subject: [PATCH] * Filemanager - clickable folders in the path --- api/js/etemplate/et2_widget_vfs.js | 128 ++++++++++++++++++++++++ api/templates/default/etemplate2.css | 35 +++++++ filemanager/js/app.js | 2 +- filemanager/templates/default/index.xet | 2 +- 4 files changed, 165 insertions(+), 2 deletions(-) diff --git a/api/js/etemplate/et2_widget_vfs.js b/api/js/etemplate/et2_widget_vfs.js index d6d7607935..c29830a293 100644 --- a/api/js/etemplate/et2_widget_vfs.js +++ b/api/js/etemplate/et2_widget_vfs.js @@ -257,6 +257,134 @@ var et2_vfsName = (function(){ "use strict"; return et2_textbox.extend( });}).call(this); et2_register_widget(et2_vfsName, ["vfs-name"]); +/** + * vfs-name + * filename automatically urlencoded on return (urldecoded on display to user) + * + * @augments et2_textbox + */ +var et2_vfsPath = (function(){ "use strict"; return et2_vfsName.extend( +{ + /** + * Constructor + * + * @memberOf et2_vfsName + */ + init: function() + { + this.div = jQuery(document.createElement("div")) + .addClass('et2_vfs'); + this.span = jQuery(document.createElement("ul")) + .appendTo(this.div); + this._super.apply(this, arguments); + }, + createInputWidget: function() + { + this._super.apply(this, arguments); + + this.div.prepend(this.input); + this.setDOMNode(this.div[0]); + + this.input.on('focus', function() { + this.input.val(this.options.value); + this.span.hide(); + }.bind(this)) + .on('focusout', function() { + // Can't use show() because it uses the wrong display + this.span.css('display', 'flex'); + this.input.val(''); + }.bind(this)); + }, + change: function(_node) + { + if(this.input.val()) + { + this.set_value(this.input.val()); + } + return this._super.apply(this, arguments); + }, + + set_value: function(_value) + { + if(_value.path) + { + _value = _value.path; + } + try + { + _value = egw.decodePath(_value); + } catch (e) + { + _value = 'Error! ' + _value; + } + if(_value === this.options.value && this._oldValue !== et2_no_init) return; + + var path_parts = _value.split('/'); + if(_value === '/') path_parts = ['']; + var path = "/"; + var text = ''; + this.span.empty().css('display', 'flex'); + this.input.val(''); + for(var i = 0; i < path_parts.length; i++) + { + path += (path=='/'?'':'/')+path_parts[i]; + text = egw.decodePath(path_parts[i]); + + var image = path=='/' ? this.egw().image('navbar','api') : this.egw().image(text); + + // Nice human-readable stuff for apps + if(path_parts[1] == 'apps') + { + if(i === 1) + { + text = this.egw().lang('applications'); + } + else if( i === 2) + { + text = this.egw().lang(path_parts[2]); + image = this.egw().image('navbar',path_parts[2].toLowerCase()); + } + else if(!isNaN(text)) + { + var link_title = this.egw().link_title(path_parts[2],path_parts[3], + function(title) { + if(!title) return; + jQuery('li',this.span).first().text(title); + }, this + ); + if(link_title && typeof link_title !== 'undefined') text = link_title; + } + + } + var self = this; + var node = jQuery(document.createElement("li")) + .addClass("vfsPath et2_clickable") + .text(text) + //.attr('title', egw.decodePath(path)) + .click({data:path, egw: this.egw()}, function(e) { + e.data.egw.open({path: e.data.data, type:'httpd/unix-directory'}, "file"); + }) + .prependTo(this.span); + if(image) + { + node.prepend(this.egw().image_element(image)); + } + jQuery(this.getDOMNode()).append(this.span); + } + + if(this.isAttached() && this.options.value !== _value) + { + this._oldValue = this.options.value; + this.options.value = _value; + this.change(); + } + }, + getValue: function() { + return egw.encodePath(this.options.value); + } +});}).call(this); +et2_register_widget(et2_vfsPath, ["vfs-path"]); + /** * vfs-name * filename automatically urlencoded on return (urldecoded on display to user) diff --git a/api/templates/default/etemplate2.css b/api/templates/default/etemplate2.css index cc98a97b3d..84e4dcd782 100644 --- a/api/templates/default/etemplate2.css +++ b/api/templates/default/etemplate2.css @@ -890,6 +890,41 @@ ul.et2_link_string { height: 16px; display: inline-block; } +div.et2_vfs { + position: relative; + overflow: hidden; +} +div.et2_vfs ul { + position: absolute; + top: 0px; + left: 0px; + padding: 0px; + margin: 1px; + margin-right: 4px; + display:flex; + justify-content: flex-start; + flex-flow: row nowrap; + /* This hides the higher level directories if overflow */ + direction: rtl; +} +div.et2_vfs li { + direction: ltr; + border-right: 1px solid silver; + display: inline-block; + padding: 4px; + padding-right: 6px; + flex: none; +} +div.et2_vfs li:hover { + background-color: green; +} +div.et2_vfs li img { + height: 16px; + float: left; + margin: 0px; + margin-left: 3px; + margin-right: 5px; +} .et2_link_list tr:hover div.delete, .et2_vfs tr:hover div.delete { visibility: visible; } diff --git a/filemanager/js/app.js b/filemanager/js/app.js index d91a406c72..96b7bdee22 100644 --- a/filemanager/js/app.js +++ b/filemanager/js/app.js @@ -112,7 +112,7 @@ app.classes.filemanager = AppJS.extend( var new_options = this.et2.getArrayMgr('sel_options').getEntry('new'); new_widget.set_select_options(new_options); } - else + else if (new_widget) { new_widget.set_disabled(true); } diff --git a/filemanager/templates/default/index.xet b/filemanager/templates/default/index.xet index 876b6dff30..2251cb6dc9 100644 --- a/filemanager/templates/default/index.xet +++ b/filemanager/templates/default/index.xet @@ -50,7 +50,7 @@