/** * EGroupware eTemplate2 - JS VFS widgets * * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @package etemplate * @subpackage api * @link https://www.egroupware.org * @author Nathan Gray * @copyright Nathan Gray 2012 */ var _a; /*egw:uses /vendor/bower-asset/jquery/dist/jquery.js; vfsSelectUI; et2_core_inputWidget; et2_core_valueWidget; et2_widget_description; et2_widget_file; expose; */ import { et2_valueWidget } from "./et2_core_valueWidget"; import { et2_createWidget, et2_register_widget } from "./et2_core_widget"; import { ClassWithAttributes } from "./et2_core_inheritance"; import { et2_textbox, et2_textbox_ro } from "./et2_widget_textbox"; import { et2_description } from "./et2_widget_description"; import { et2_selectAccount_ro } from "./et2_widget_selectAccount"; import { et2_file } from "./et2_widget_file"; import { et2_dialog } from "./et2_widget_dialog"; import { et2_inputWidget } from "./et2_core_inputWidget"; import { et2_no_init } from "./et2_core_common"; import { egw, egw_get_file_editor_prefered_mimes } from "../jsapi/egw_global"; import { expose } from "./expose"; import { egw_getAppObjectManager } from "../egw_action/egw_action.js"; import { EGW_KEY_ENTER, egw_keyHandler } from '../egw_action/egw_keymanager.js'; import { etemplate2 } from "./etemplate2"; /** * Class which implements the "vfs" XET-Tag * * @augments et2_valueWidget */ export class et2_vfs extends et2_valueWidget { /** * Constructor * * @memberOf et2_vfs */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfs._attributes, _child || {})); this.span = null; this.value = ""; this.span = jQuery(document.createElement("ul")) .addClass('et2_vfs'); this.setDOMNode(this.span[0]); } getValue() { return this.value; } set_value(_value) { if (typeof _value !== 'object') { // Only warn if it's an actual value, just blank for falsy values if (_value) { this.egw().debug("warn", "%s only has path, needs full array", this.id, _value); } this.span.empty().text(_value); return; } this.span.empty(); this.value = _value; let path = _value.path ? _value.path : '/'; // calculate path as parent of name, which can contain slashes // eg. _value.path=/home/ralf/sub/file, _value.name=sub/file --> path=/home/ralf // --> generate clickable fields for sub/ + file let sub_path = path.substring(0, _value.path.length - _value.name.length - 1); let path_offset, path_parts; if (_value.path.indexOf(_value.name) >= 0 && sub_path[sub_path.length - 1] === '/') { path = sub_path; path_offset = path.split('/').length; path_parts = _value.path.split('/'); } else { if (_value.path.indexOf(_value.name) >= 0) { // Remove name from end, so we can add it again later path = sub_path; } path_offset = 0; path_parts = _value.name.split('/'); } let text; for (let i = path_offset; i < path_parts.length; i++) { path += (path == '/' ? '' : '/') + path_parts[i]; text = egw.decodePath(path_parts[i]); // Nice human-readable stuff for apps if (path_parts[1] == 'apps') { switch (path_parts.length) { case 2: if (i == 1) { text = this.egw().lang('applications'); } break; case 3: if (i == 2) { text = this.egw().lang(path_parts[2]); } break; case 4: if (!isNaN(text)) { let link_title = this.egw().link_title(path_parts[2], path_parts[3], function (title) { if (!title || this.value.name == title) return; jQuery('li', this.span).last().text(title); }, this); if (link_title && typeof link_title !== 'undefined') text = link_title; } break; } } let self = this; var data = { path: path, type: i < path_parts.length - 1 ? et2_vfs.DIR_MIME_TYPE : _value.mime }; var node = jQuery(document.createElement("li")) .addClass("vfsFilename") .text(text + (i < path_parts.length - 1 ? '/' : '')) //.attr('title', egw.decodePath(path)) .addClass("et2_clickable et2_link") .click({ data: data, egw: this.egw() }, function (e) { if (!self.onclick) { e.data.egw.open(e.data.data, "file"); } else if (self.click(e)) { e.data.egw.open(e.data.data, "file"); } }) .appendTo(this.span); } // Last part of path do default action this._bind_default_action(node, data); } _bind_default_action(node, data) { let links = []; let widget = this; let defaultAction = null; let object = null; let app = this.getInstanceManager().app; while (links.length === 0 && widget.getParent()) { object = egw_getAppObjectManager(app).getObjectById(widget.id); if (object && object.manager && object.manager.children) { links = object.manager.children; } widget = widget.getParent(); } for (let k in links) { if (links[k].default && links[k].enabled.exec(links[k])) { defaultAction = links[k]; break; } } if (defaultAction && !this.onclick) { node.off('click').click({ data: data, egw: this.egw() }, function (e) { // Wait until object selection happens window.setTimeout(function () { // execute default action egw_keyHandler(EGW_KEY_ENTER, false, false, false); }); // Select row return true; }.bind({ data: data, object: object })); } } /** * Code for implementing et2_IDetachedDOM (data grid) * * @param {array} _attrs array of attribute-names to push further names onto */ getDetachedAttributes(_attrs) { _attrs.push("value"); } getDetachedNodes() { return [this.span[0]]; } setDetachedAttributes(_nodes, _values) { this.span = jQuery(_nodes[0]); if (typeof _values["value"] != 'undefined') { this.set_value(_values["value"]); } } } et2_vfs._attributes = { "value": { "type": "any", "description": "Array of (stat) information about the file" } }; /** * Mime type of directories */ et2_vfs.DIR_MIME_TYPE = 'httpd/unix-directory'; et2_register_widget(et2_vfs, ["vfs"]); /** * vfs-name * filename automatically urlencoded on return (urldecoded on display to user) * * @augments et2_textbox */ export class et2_vfsName extends et2_textbox { /** * Constructor * * @memberOf et2_vfsName */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsName._attributes, _child || {})); this.input.addClass("et2_vfs"); } set_value(_value) { if (_value.path) { _value = _value.path; } try { _value = egw.decodePath(_value); } catch (e) { _value = 'Error! ' + _value; } super.set_value(_value); } getValue() { return egw.encodePath(super.getValue() || ''); } } et2_register_widget(et2_vfsName, ["vfs-name"]); /** * vfs-name * filename automatically urlencoded on return (urldecoded on display to user) * * @augments et2_textbox */ export class et2_vfsPath extends et2_vfsName { /** * Constructor * * @memberOf et2_vfsName */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsPath._attributes, _child || {})); } createInputWidget() { super.createInputWidget(); this.div = jQuery(document.createElement("div")) .addClass('et2_vfsPath'); this.span = jQuery(document.createElement("ul")) .appendTo(this.div); this.div.prepend(this.input); this.setDOMNode(this.div[0]); this.span.on('wheel', function (e) { var delta = e.originalEvent["deltaY"] > 0 ? 30 : -30; this.scrollLeft = this.scrollLeft - delta; }); this.span.on('mouseover', function (e) { if (this.scrollWidth > this.clientWidth) { jQuery(this).addClass('scrollable'); } else { jQuery(this).removeClass('scrollable'); } }); 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(_node) { if (this.input.val()) { this.set_value(this.input.val()); } return super.change(_node); } set_value(_value) { if (_value.path) { _value = _value.path; } if (_value === this.options.value && this._oldValue !== et2_no_init) return; let path_parts = _value.split('/'); if (_value === '/') path_parts = ['']; let path = "/"; let text = ''; if (this.span) this.span.empty().css('display', 'flex'); this.input.val(''); for (let i = 0; i < path_parts.length; i++) { path += (path == '/' ? '' : '/') + path_parts[i]; text = egw.decodePath(path_parts[i]); let 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)) { let 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; } } let self = this; let node = jQuery(document.createElement("li")) .addClass("vfsPath et2_clickable") .text(text) //.attr('title', egw.decodePath(path)) .click({ data: path, egw: this.egw() }, function (e) { return self.set_value(e.data.data); }) .prependTo(this.span); if (image && !this.options.noicon) { 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() { return this.options ? this.options.value : null; } } et2_vfsPath._attributes = { noicon: { type: "boolean", description: "suppress folder icons", default: true } }; et2_register_widget(et2_vfsPath, ["vfs-path"]); /** * vfs-name * filename automatically urlencoded on return (urldecoded on display to user) * * @augments et2_textbox_ro */ export class et2_vfsName_ro extends et2_textbox_ro { /** * Constructor * * @memberOf et2_vfsName_ro */ /** * Constructor */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsName_ro._attributes, _child || {})); } set_value(_value) { if (_value.path) { _value = _value.path; } try { _value = egw.decodePath(_value); } catch (e) { _value = 'Error! ' + _value; } super.set_value(_value); } getValue() { return egw.encodePath(super.getValue() || ''); } } et2_register_widget(et2_vfsName_ro, ["vfs-name_ro"]); /** * vfs-mime: icon for mimetype of file, or thumbnail * incl. optional link overlay icon, if file is a symlink * * Creates following structure * * * * * span.overlayContainer is optional and only generated for symlinks * @augments et2_valueWidget */ export class et2_vfsMime extends expose((_a = class et2_vfsMime extends et2_valueWidget { /** * Constructor * * @memberOf et2_vfsMime */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsMime._attributes, _child || {})); this.iconOverlayContainer = null; this.image = null; this.iconOverlayContainer = jQuery(document.createElement('span')).addClass('iconOverlayContainer'); this.image = jQuery(document.createElement("img")); this.image.addClass("et2_vfs vfsMimeIcon"); this.iconOverlayContainer.append(this.image); this.setDOMNode(this.iconOverlayContainer[0]); } /** * Handler for expose slide action, from expose * Returns data needed for the given index, or false to let expose handle it * * @param {Gallery} gallery * @param {integer} index * @param {DOMNode} slide * @return {Array} array of objects consist of media contnet */ expose_onslide(gallery, index, slide) { var content = false; if (this.options.expose_callback && typeof this.options.expose_callback == 'function') { //Call the callback to load more items content = this.options.expose_callback.call(this, [gallery, index]); if (content) this.add(content); } return content; } /** * Function to get media content to feed the expose * * @param {type} _value * @returns {Array} return an array of object consists of media content */ getMedia(_value) { let base_url = egw.webserverUrl.match(/^\/ig/) ? egw(window).window.location.origin + egw.webserverUrl : egw.webserverUrl; let mediaContent = [{ title: _value.name, type: _value.mime, href: _value.download_url }]; // check if download_url is not already an url (some stream-wrappers allow to specify that!) if (_value.download_url && (_value.download_url[0] == '/' || _value.download_url.substr(0, 4) != 'http')) { mediaContent[0].href = base_url + _value.download_url; if (mediaContent[0].href && mediaContent[0].href.match(/\/webdav.php/, 'ig')) { mediaContent[0]["download_href"] = mediaContent[0].href + '?download'; } } if (_value && _value.mime && _value.mime.match(/video\//, 'ig')) { mediaContent[0]["thumbnail"] = this.egw().mime_icon(_value.mime, _value.path, undefined, _value.mtime); } else { mediaContent[0]["thumbnail"] = _value.path && _value.mime ? this.egw().mime_icon(_value.mime, _value.path, undefined, _value.mtime) : this.image.attr('src') + '&thheight=128'; } return mediaContent; } set_value(_value) { if (typeof _value !== 'object') { this.egw().debug("warn", "%s only has path, needs array with path & mime", this.id, _value); // Keep going, will be 'unknown type' } let src = this.egw().mime_icon(_value.mime, _value.path, undefined, _value.mtime); if (src) { // Set size of thumbnail if (src.indexOf("thumbnail.php") > -1) { if (this.options.size) { src += "&thsize=" + this.options.size; } else if (this.options.thumb_mime_size) { let mime_size = this.options.thumb_mime_size.split(','); let mime_regex = RegExp(_value.mime.split('/')[0]); if (typeof mime_size != 'undefined' && jQuery.isArray(mime_size) && !isNaN(mime_size[mime_size.length - 1]) && isNaN(mime_size[0]) && this.options.thumb_mime_size.match(mime_regex[0], 'ig')) { src += "&thsize=" + mime_size[mime_size.length - 1]; } } this.image.css("max-width", "100%"); } this.image.attr("src", src); // tooltip for mimetypes with available detailed thumbnail if (_value.mime && _value.mime.match(/application\/vnd\.oasis\.opendocument\.(text|presentation|spreadsheet|chart)/)) { let tooltip_target = this.image.parent().parent().parent().length > 0 ? // Nextmatch row this.image.parent().parent().parent() : // Not in nextmatch this.image.parent(); tooltip_target.tooltip({ items: "img", position: { my: "right top", at: "left top", collision: "flipfit" }, content: function () { return ''; } }); } } // add/remove link icon, if file is (not) a symlink if ((_value.mode & et2_vfsMode.types.l) == et2_vfsMode.types.l) { if (typeof this.overlayContainer == 'undefined') { this.overlayContainer = jQuery(document.createElement('span')).addClass('overlayContainer'); this.overlayContainer.append(jQuery(document.createElement('img')) .addClass('overlay').attr('src', this.egw().image('link', 'etemplate'))); this.iconOverlayContainer.append(this.overlayContainer); } } else if (typeof this.overlayContainer != 'undefined') { this.overlayContainer.remove(); delete this.overlayContainer; } } /** * Implementation of "et2_IDetachedDOM" for fast viewing in gridview * Override to add needed attributes * * @param {array} _attrs array of attribute-names to push further names onto */ getDetachedAttributes(_attrs) { _attrs.push("value", "class"); } getDetachedNodes() { return [this.node, this.iconOverlayContainer[0], this.image[0]]; } setDetachedAttributes(_nodes, _values) { this.iconOverlayContainer = jQuery(_nodes[1]); this.image = jQuery(_nodes[2]); this.node = _nodes[0]; this.overlayContainer = _nodes[0].children[1]; if (typeof _values['class'] != "undefined") { this.image.addClass(_values['class']); } if (typeof _values['value'] != "undefined") { this.set_value(_values['value']); } } }, _a._attributes = { "value": { "type": "any", "description": "Array of (stat) information about the file" }, "size": { "name": "Icon size", "type": "integer", "description": "Size of icon / thumbnail, in pixels", "default": et2_no_init }, "expose_callback": { "name": "expose_callback", "type": "js", "default": et2_no_init, "description": "JS code which is executed when expose slides." }, expose_view: { name: "Expose view", type: "boolean", default: true, description: "Clicking on an image would popup an expose view" }, thumb_mime_size: { name: "Image thumbnail size", type: "string", default: "", description: " Size of thumbnail in pixel for specified mime type with syntax of: mime_type(s),size (eg. image,video,128)" } }, _a.legacyOptions = ["size"], _a)) { } ; et2_register_widget(et2_vfsMime, ["vfs-mime"]); /** * vfs-size * Human readable file sizes * * @augments et2_description */ export class et2_vfsSize extends et2_description { /** * Constructor * * @memberOf et2_vfsSize */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsSize._attributes, _child || {})); this.span.addClass("et2_vfs"); } human_size(size) { if (typeof size !== "number") { size = parseInt(size); } if (!size) { size = 0; } const units = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']; let i = 0; while (size >= 1024) { size /= 1024; ++i; } return size.toFixed(i == 0 ? 0 : 1) + ' ' + units[i]; } set_value(_value) { if (_value.size) { _value = _value.size; } jQuery(this.node).text(this.human_size(_value)); } setDetachedAttributes(_nodes, _values) { if (typeof _values["value"] !== "undefined") { this.node = _nodes[0]; this.set_value(_values["value"]); delete _values["value"]; } super.setDetachedAttributes(_nodes, _values); } } et2_vfsSize._attributes = { "value": { "type": "integer" } }; et2_register_widget(et2_vfsSize, ["vfs-size"]); /** * vfs-mode: textual representation of permissions + extra bits * * @augments et2_description */ export class et2_vfsMode extends et2_description { /** * Constructor * * @memberOf et2_vfsMode */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsMode._attributes, _child || {})); this.span.addClass("et2_vfs"); } /** * Get text for file stuff * Result will be like -rwxr--r--. First char is type, then read, write, execute (or other bits) for * user, group, world * * @param {number} _value vfs mode */ text_mode(_value) { let text = []; if (typeof _value != "number") { _value = parseInt(_value); } if (!_value) return "----------"; // Figure out type let type = 'u'; // unknown for (let flag in et2_vfsMode.types) { if ((_value & et2_vfsMode.types[flag]) == et2_vfsMode.types[flag]) { type = flag; break; } } // World, group, user - build string backwards for (let i = 0; i < 3; i++) { for (let perm in et2_vfsMode.perms) { if (_value & et2_vfsMode.perms[perm]) { text.unshift(perm); } else { text.unshift("-"); } } _value = _value >> 3; } // Sticky / UID / GID for (let i = 0; i < et2_vfsMode.sticky.length; i++) { if (et2_vfsMode.sticky[i].mask & _value) { let current = text[et2_vfsMode.sticky[i].position]; text[et2_vfsMode.sticky[i].position] = et2_vfsMode.sticky[i]["char"]; if (current == 'x') text[et2_vfsMode.sticky[i].position].toLowerCase(); } } return type + text.join(''); } set_value(_value) { if (_value.size) { _value = _value.size; } let text = this.text_mode(_value); jQuery(this.node).text(text); } setDetachedAttributes(_nodes, _values) { if (typeof _values["value"] !== "undefined") { this.node = _nodes[0]; this.set_value(_values["value"]); delete _values["value"]; } super.setDetachedAttributes(_nodes, _values); } } // Masks for file types et2_vfsMode.types = { 'l': 0xA000, 's': 0xC000, 'p': 0x1000, 'c': 0x2000, 'd': 0x4000, 'b': 0x6000, '-': 0x8000 // Regular }; // Sticky / UID / GID et2_vfsMode.sticky = [ { mask: 0x200, "char": "T", position: 9 }, { mask: 0x400, "char": "S", position: 6 }, { mask: 0x800, "char": "S", position: 3 } // SUID ]; et2_vfsMode.perms = { 'x': 0x1, 'w': 0x2, 'r': 0x4 // Read }; et2_register_widget(et2_vfsMode, ["vfs-mode"]); /** * vfs-uid / vfs-gid: Displays the name for an ID. * Same as read-only selectAccount, except if there's no user it shows "root" * * @augments et2_selectAccount_ro */ export class et2_vfsUid extends et2_selectAccount_ro { /** * @memberOf et2_vfsUid * @param _node * @param _value */ set_title(_node, _value) { if (_value == "") { arguments[1] = "root"; } super.set_title(_node, _value); } } et2_register_widget(et2_vfsUid, ["vfs-uid", "vfs-gid"]); /* vfs-upload aka VFS file: displays either download and delete (x) links or a file upload * + ID is either a vfs path or colon separated $app:$id:$relative_path, eg: infolog:123:special/offer * + if empty($id) / new entry, file is created in a hidden temporary directory in users home directory * and calling app is responsible to move content of that dir to entry directory, after entry is saved * + option: required mimetype or regular expression for mimetype to match, eg. '/^text\//i' for all text files * + if path ends in a slash, multiple files can be uploaded, their original filename is kept then * * @augments et2_file */ export class et2_vfsUpload extends et2_file { /** * Constructor * * @param _parent * @param attrs * @memberof et2_vfsUpload */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsUpload._attributes, _child || {})); this.list = null; jQuery(this.node).addClass("et2_vfs"); if (!this.options.path) { this.options.path = this.options.id; } // If the path is a directory, allow multiple uploads if (this.options.path.substr(-1) == '/') { this.set_multiple(true); } this.list = jQuery(document.createElement('table')).appendTo(this.node); } /** * Get any specific async upload options */ getAsyncOptions(self) { return jQuery.extend({}, super.getAsyncOptions(self), { target: egw.ajaxUrl("EGroupware\\Api\\Etemplate\\Widget\\Vfs::ajax_upload") }); } /** * If there is a file / files in the specified location, display them * Value is the information for the file[s] in the specified location. * * @param {Object{}} _value */ set_value(_value) { // Remove previous while (this._children.length > 0) { var node = this._children[this._children.length - 1]; this.removeChild(node); node.destroy(); } this.progress.empty(); this.list.empty(); // Set new if (typeof _value == 'object' && _value && Object.keys(_value).length) { for (let i in _value) { this._addFile(_value[i]); } } return true; } getDOMNode(sender) { if (sender && sender !== this && sender._type.indexOf('vfs') >= 0) { var value = sender.getValue && sender.getValue() || sender.options.value || {}; var row = jQuery("[data-path='" + (value.path.replace(/'/g, '"')) + "']", this.list); if (sender._type === 'vfs-mime') { return jQuery('.icon', row).get(0) || null; } else { return jQuery('.title', row).get(0) || null; } } else { return super.getDOMNode(sender); } } /** * Add in the request id * * @param {type} form */ beforeSend(form) { let extra = super.beforeSend(form); extra["path"] = this.options.path; return extra; } /** * A file upload is finished, update the UI * * @param {object} file * @param {string|object} response */ finishUpload(file, response) { let result = super.finishUpload(file, response); if (typeof response == 'string') response = jQuery.parseJSON(response); if (response.response[0] && typeof response.response[0].data.length == 'undefined') { for (let key in response.response[0].data) { let value = response.response[0].data[key]; if (value && value.path) { this._addFile(value); jQuery("[data-file='" + file.fileName.replace(/'/g, '"') + "']", this.progress).hide(); } } } return result; } _addFile(file_data) { if (jQuery("[data-path='" + file_data.path.replace(/'/g, '"') + "']").remove().length) { for (var child_index = this._children.length - 1; child_index >= 0; child_index--) { var child = this._children[child_index]; if (child.options.value.path === file_data.path) { this.removeChild(child); child.destroy(); } } } // Set up for expose if (file_data && typeof file_data.download_url === "undefined") { file_data.download_url = "/webdav.php" + file_data.path; } let row = jQuery(document.createElement("tr")) .attr("data-path", file_data.path.replace(/'/g, '"')) .attr("draggable", "true") .appendTo(this.list); jQuery(document.createElement("td")) .addClass('icon') .appendTo(row); jQuery(document.createElement("td")) .addClass('title') .appendTo(row); let mime = et2_createWidget('vfs-mime', { value: file_data }, this); // Trigger expose on click, if supported let vfs_attrs = { value: file_data, onclick: undefined }; if (file_data && (typeof file_data.download_url != 'undefined')) { var fe_mime = egw_get_file_editor_prefered_mimes(file_data.mime); // Check if the link entry is mime with media type, in order to open it in expose view if (typeof file_data.mime === 'string' && (file_data.mime.match(mime.mime_regexp, 'ig') || (fe_mime && fe_mime.mime[file_data.mime]))) { vfs_attrs.onclick = function (ev) { ev.stopPropagation(); // Pass it off to the associated vfsMime widget jQuery('img', this.parentNode.parentNode).trigger("click"); return false; }; } } let vfs = et2_createWidget('vfs', vfs_attrs, this); // If already attached, need to do this explicitly if (this.isAttached()) { mime.set_value(file_data); vfs.set_value(file_data); mime.doLoadingFinished(); vfs.doLoadingFinished(); } // Add in delete button if (!this.options.readonly) { let self = this; let delete_button = jQuery(document.createElement("td")) .appendTo(row); jQuery(" *
") .appendTo(delete_button) // We don't use ui-icon because it assigns a bg image .addClass("delete icon") .bind('click', function () { et2_createWidget("dialog", { callback: function (button) { if (button == et2_dialog.YES_BUTTON) { egw.json("filemanager_ui::ajax_action", [ 'delete', [row.attr('data-path').replace(/"/g, "'")], '' ], function (data) { if (data && data.errs == 0) { row.slideUp(null, row.remove); } if (data && data.msg) { self.egw().message(data.msg, data.errs == 0 ? 'success' : 'error'); } }).sendRequest(); } }, message: self.egw().lang('Delete file') + '?', title: self.egw().lang('Confirmation required'), buttons: et2_dialog.BUTTONS_YES_NO, dialog_type: et2_dialog.QUESTION_MESSAGE, width: 250 }, self); }); } } } et2_vfsUpload._attributes = { "value": { "type": "any" // Either nothing, or an object with file info }, "path": { "name": "Path", "description": "Upload files to the specified VFS path", "type": "string", "default": '' } }; et2_vfsUpload.legacyOptions = ["mime"]; et2_register_widget(et2_vfsUpload, ["vfs-upload"]); export class et2_vfsSelect extends et2_inputWidget { /** * Constructor * * @param _parent * @param _attrs * @memberOf et2_vfsSelect */ constructor(_parent, _attrs, _child) { // Call the inherited constructor super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_vfsSelect._attributes, _child || {})); // Allowed mode options this.modes = ['open', 'open-multiple', 'saveas', 'select-dir']; // Allow no child widgets this.supportedWidgetClasses = []; this.button = jQuery(document.createElement("button")) .attr("title", this.egw().lang("Select file(s) from VFS")) .addClass("et2_button et2_vfs_btn") .css("background-image", "url(" + this.egw().image("filemanager/navbar") + ")"); if (this.options.readonly) { this.button.hide(); } if (this.options.button_caption != "") { this.button.text(this.options.button_caption); } this.setDOMNode(this.button[0]); } _content(_content, _callback) { egw(window).loading_prompt('vfs-select', true, '', 'body'); let self = this; if (typeof app.vfsSelectUI != "undefined") { if (this.dialog && this.dialog.div) this.dialog.div.dialog('close'); delete app.vfsSelectUI; } let attrs = { mode: this.options.mode, label: this.options.button_label, path: this.options.path || null, mime: this.options.mime || null, name: this.options.name, method: this.options.method, recentPaths: et2_vfsSelect._getRecentPaths() }; var callback = _callback || this._buildDialog; egw(window).json('EGroupware\\Api\\Etemplate\\Widget\\Vfs::ajax_vfsSelect_content', [_content, attrs], function (_content) { egw(window).loading_prompt('vfs-select', false); callback.apply(self, arguments); }).sendRequest(true); } /** * Builds file navigator dialog * * @param {object} _data content */ _buildDialog(_data) { if (!_data.content.mode.match(/open|open-multiple|saveas|select-dir/)) { egw.debug('warn', 'Mode is not matched!'); return; } let self = this; let buttons = [ { text: egw.lang(_data.content.label), id: "submit", image: _data.content.mode.match(/saveas|select-dir/) ? "save" : this.options.button_icon } ]; let extra_buttons_action = {}; if (this.options.extra_buttons && this.options.method) { for (let i = 0; i < this.options.extra_buttons.length; i++) { delete (this.options.extra_buttons[i]['click']); buttons.push(this.options.extra_buttons[i]); extra_buttons_action[this.options.extra_buttons[i]['id']] = this.options.extra_buttons[i]['id']; } } buttons.push({ text: egw.lang("Close"), id: "close", image: "cancel" }); // Don't rely only on app_name to fetch et2 object as app_name may not // always represent current app of the window, e.g.: mail admin account. // Try to fetch et2 from its template name. let etemplate = jQuery('form').data('etemplate'); let et2; let currentApp = egw(window).app_name(); if (etemplate && etemplate.name && !app[currentApp]) { et2 = etemplate2.getByTemplate(etemplate.name)[0]; currentApp = et2.name.split('.')[0]; } else { et2 = etemplate2.getByApplication(currentApp)[0]; } let data = jQuery.extend(_data, { 'currentapp': currentApp, etemplate_exec_id: et2.etemplate_exec_id }); // define a mini app object for vfs select UI app.vfsSelectUI = new app.classes.vfsSelectUI; // callback for dialog this.submit_callback = function (submit_button_id, submit_value, savemode) { if ((submit_button_id == 'submit' || (extra_buttons_action && extra_buttons_action[submit_button_id])) && submit_value) { let files = []; switch (_data.content.mode) { case 'open-multiple': if (submit_value.dir && submit_value.dir.selected) { for (var key in Object.keys(submit_value.dir.selected)) { if (submit_value.dir.selected[key] != "") { files.push(submit_value.path + '/' + submit_value.dir.selected[key]); } } } break; case 'select-dir': files = submit_value.path; break; default: if (self.options.method === 'download') submit_value.path = _data.content.download_baseUrl; files = submit_value.path + '/' + submit_value.name; if (self.options.mode === 'saveas' && !savemode) { for (var p in _data.content.dir) { if (_data.content.dir[p]['name'] == submit_value.name) { var saveModeDialogButtons = [ { text: self.egw().lang("Yes"), id: "overwrite", class: "ui-priority-primary", "default": true, image: 'check' }, { text: self.egw().lang("Rename"), id: "rename", image: 'edit' }, { text: self.egw().lang("Cancel"), id: "cancel" } ]; return et2_dialog.show_prompt(function (_button_id, _value) { switch (_button_id) { case "overwrite": return self.submit_callback(submit_button_id, submit_value, 'overwrite'); case "rename": submit_value.name = _value; return self.submit_callback(submit_button_id, submit_value, 'rename'); } }, self.egw().lang('Do you want to overwrite existing file %1 in directory %2?', submit_value.name, submit_value.path), self.egw().lang('File %1 already exists', submit_value.name), submit_value.name, saveModeDialogButtons, null); } } } break; } et2_vfsSelect._setRecentPaths(submit_value.path); self.value = files; if (self.options.method && self.options.method !== 'download') { egw(window).request(self.options.method, [self.options.method_id, files, submit_button_id, savemode]).then(function (data) { jQuery(self.node).change(); }); } else { jQuery(self.node).change(); } delete app.vfsSelectUI; return true; } }; this.dialog = et2_createWidget("dialog", { callback: this.submit_callback, title: this.options.dialog_title, buttons: buttons, minWidth: 500, minHeight: 400, width: 400, value: data, template: egw.webserverUrl + '/api/templates/default/vfsSelectUI.xet?1', resizable: false }, et2_dialog._create_parent('api')); this.dialog.template.uniqueId = 'api.vfsSelectUI'; app.vfsSelectUI.et2 = this.dialog.template.widgetContainer; app.vfsSelectUI.vfsSelectWidget = this; // Keep the dialog always at the top this.dialog.div.parent().css({ "z-index": 100000 }); this.dialog.div.on('load', function (e) { app.vfsSelectUI.et2_ready(app.vfsSelectUI.et2, 'api.vfsSelectUI'); }); // we need an etemplate_exec_id for better handling serverside parts of // widgets and since we can not have a etemplate_exec_id specifically // for dialog template our best shot is to inherit its parent etemplate_exec_id. this.dialog.template.etemplate_exec_id = et2.etemplate_exec_id; } /** * Set recent path into sessionStorage * @param {string} _path */ static _setRecentPaths(_path) { let recentPaths = egw.getSessionItem('api', 'vfsRecentPaths') ? egw.getSessionItem('api', 'vfsRecentPaths').split(',') : []; if (recentPaths.indexOf(_path) == -1) recentPaths.push(_path); egw.setSessionItem('api', 'vfsRecentPaths', recentPaths); } /** * Get recent paths from sessionStorage * @returns {Array} returns an array of recent paths */ static _getRecentPaths() { return egw.getSessionItem('api', 'vfsRecentPaths') ? egw.getSessionItem('api', 'vfsRecentPaths').split(',') : []; } /** * click handler * @param {event object} e */ click(e) { this._content.call(this, null); } /** * Set the dialog's mode. * Valid options are in et2_vfsSelect.modes * * @param {string} mode 'open', 'open-multiple', 'saveas' or 'select-dir' */ set_mode(mode) { // Check mode if (jQuery.inArray(mode, this.modes) < 0) { this.egw().debug("warn", "Invalid mode for '%s': %s Valid options:", this.id, mode, this.modes); return; } this.options.mode = mode; } /** * Set the label on the dialog's OK button. * * @param {string} label */ set_button_label(label) { this.options.button_label = label; } /** * Set the caption for vfs-select button * * @param {string} caption string value as a caption */ set_button_caption(caption) { this.options.button_caption = caption; } /** * Set the ID passed to the server side callback * * @param {string} id */ set_method_id(id) { this.options.method_id = id; } set_readonly(readonly) { this.options.readonly = Boolean(readonly); if (this.options.readonly) { this.button.hide(); } else { this.button.show(); } } set_value(value) { this.value = value; } getValue() { return this.value; } } et2_vfsSelect._attributes = { "mode": { name: "Dialog mode", type: "string", description: "One of {open|open-multiple|saveas|select-dir}", default: "open-multiple" }, "method": { name: "Server side callback", type: "string", description: "Server side callback to process selected value(s) in \n\ app.class.method or class::method format. The first parameter will \n\ be Method ID, the second the file list. 'download' is reserved and it \n\ means it should use download_baseUrl instead of path in value (no method\n\ will be actually executed)." }, "method_id": { name: "Method ID", type: "any", description: "optional parameter passed to server side callback.\n\ Can be a string or a function.", default: "" }, "path": { name: "Path", type: "string", description: "Start path in VFS. Leave unset to use the last used path." }, "mime": { name: "Mime type", type: "any", description: "Limit display to the given mime-type" }, "button_label": { name: "Button label", description: "Set the label on the dialog's OK button.", default: "open" }, "value": { type: "any", description: "Array of paths (strings)" }, "button_caption": { name: "button caption", type: "string", default: "Select files from Filemanager ...", description: "Caption for vfs-select button.", translate: true }, "button_icon": { name: "button icon", type: "string", default: "check", description: "Custom icon to show on submit button." }, "name": { name: "File name", type: "any", description: "file name", default: "" }, "dialog_title": { name: "dialog title", type: "string", default: "Save as", description: "Title of dialog", translate: true }, "extra_buttons": { name: "extra action buttons", type: "any", description: "Extra buttons passed to dialog. It's co-related to method." } }; ; et2_register_widget(et2_vfsSelect, ["vfs-select"]); //# sourceMappingURL=et2_widget_vfs.js.map