devextreme
Version:
HTML5 JavaScript Component Suite for Responsive Web Development
294 lines (292 loc) • 12.4 kB
JavaScript
/**
* DevExtreme (cjs/ui/file_manager/ui.file_manager.files_tree_view.js)
* Version: 24.2.6
* Build date: Mon Mar 17 2025
*
* Copyright (c) 2012 - 2025 Developer Express Inc. ALL RIGHTS RESERVED
* Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/
*/
"use strict";
exports.default = void 0;
var _renderer = _interopRequireDefault(require("../../core/renderer"));
var _extend = require("../../core/utils/extend");
var _icon = require("../../core/utils/icon");
var _common = require("../../core/utils/common");
var _ui = _interopRequireDefault(require("../widget/ui.widget"));
var _tree_view = _interopRequireDefault(require("../tree_view"));
var _uiFile_manager = _interopRequireDefault(require("./ui.file_manager.file_actions_button"));
var _deferred = require("../../core/utils/deferred");
var _window = require("../../core/utils/window");
var _type = require("../../core/utils/type");
function _interopRequireDefault(e) {
return e && e.__esModule ? e : {
default: e
}
}
const FILE_MANAGER_DIRS_TREE_CLASS = "dx-filemanager-dirs-tree";
const FILE_MANAGER_DIRS_TREE_FOCUSED_ITEM_CLASS = "dx-filemanager-focused-item";
const FILE_MANAGER_DIRS_TREE_ITEM_TEXT_CLASS = "dx-filemanager-dirs-tree-item-text";
const TREE_VIEW_ITEM_CLASS = "dx-treeview-item";
class FileManagerFilesTreeView extends _ui.default {
_initMarkup() {
this._initActions();
this._getCurrentDirectory = this.option("getCurrentDirectory");
this._createFileActionsButton = _common.noop;
this._storeExpandedState = this.option("storeExpandedState") || false;
const $treeView = (0, _renderer.default)("<div>").addClass("dx-filemanager-dirs-tree").appendTo(this.$element());
const treeViewOptions = {
dataStructure: "plain",
rootValue: "",
createChildren: this._onFilesTreeViewCreateSubDirectories.bind(this),
itemTemplate: this._createFilesTreeViewItemTemplate.bind(this),
keyExpr: "getInternalKey",
parentIdExpr: "parentDirectory.getInternalKey",
displayExpr: itemInfo => itemInfo.getDisplayName(),
hasItemsExpr: "fileItem.hasSubDirectories",
onItemClick: e => this._actions.onDirectoryClick(e),
onItemExpanded: e => this._onFilesTreeViewItemExpanded(e),
onItemCollapsed: e => this._onFilesTreeViewItemCollapsed(e),
onItemRendered: e => this._onFilesTreeViewItemRendered(e),
onContentReady: () => this._actions.onFilesTreeViewContentReady()
};
if (this._contextMenu) {
this._contextMenu.option("onContextMenuHidden", (() => this._onContextMenuHidden()));
treeViewOptions.onItemContextMenu = e => this._onFilesTreeViewItemContextMenu(e);
this._createFileActionsButton = (element, options) => this._createComponent(element, _uiFile_manager.default, options)
}
this._filesTreeView = this._createComponent($treeView, _tree_view.default, treeViewOptions)
}
_initActions() {
this._actions = {
onDirectoryClick: this._createActionByOption("onDirectoryClick"),
onFilesTreeViewContentReady: this._createActionByOption("onFilesTreeViewContentReady")
}
}
_render() {
super._render();
const that = this;
setTimeout((() => {
that._updateFocusedElement()
}))
}
_onFilesTreeViewCreateSubDirectories(rootItem) {
const getDirectories = this.option("getDirectories");
const directoryInfo = rootItem && rootItem.itemData || null;
return getDirectories && getDirectories(directoryInfo, true)
}
_onFilesTreeViewItemRendered(_ref) {
let {
itemData: itemData
} = _ref;
const currentDirectory = this._getCurrentDirectory();
if (currentDirectory && currentDirectory.fileItem.equals(itemData.fileItem)) {
this._updateFocusedElement();
this._restoreScrollTopPosition()
}
}
_onFilesTreeViewItemExpanded(_ref2) {
let {
itemData: itemData
} = _ref2;
if (this._storeExpandedState) {
itemData.expanded = true
}
}
_onFilesTreeViewItemCollapsed(_ref3) {
let {
itemData: itemData
} = _ref3;
if (this._storeExpandedState) {
itemData.expanded = false
}
}
_createFilesTreeViewItemTemplate(itemData, itemIndex, itemElement) {
const $itemElement = (0, _renderer.default)(itemElement);
const $itemWrapper = $itemElement.closest(this._filesTreeViewItemSelector);
$itemWrapper.data("item", itemData);
const $image = (0, _icon.getImageContainer)(itemData.icon);
const $text = (0, _renderer.default)("<span>").text(itemData.getDisplayName()).addClass("dx-filemanager-dirs-tree-item-text");
const $button = (0, _renderer.default)("<div>");
$itemElement.append($image, $text, $button);
this._createFileActionsButton($button, {
onClick: e => this._onFileItemActionButtonClick(e)
})
}
_onFilesTreeViewItemContextMenu(_ref4) {
let {
itemElement: itemElement,
event: event
} = _ref4;
event.preventDefault();
event.stopPropagation();
const itemData = (0, _renderer.default)(itemElement).data("item");
this._contextMenu.showAt([itemData], itemElement, event, {
itemData: itemData,
itemElement: itemElement
})
}
_onFileItemActionButtonClick(_ref5) {
let {
component: component,
element: element,
event: event
} = _ref5;
event.stopPropagation();
const itemElement = component.$element().closest(this._filesTreeViewItemSelector);
const itemData = itemElement.data("item");
const target = {
itemData: itemData,
itemElement: itemElement,
isActionButton: true
};
this._contextMenu.showAt([itemData], element, event, target);
this._activeFileActionsButton = component;
this._activeFileActionsButton.setActive(true)
}
_onContextMenuHidden() {
if (this._activeFileActionsButton) {
this._activeFileActionsButton.setActive(false)
}
}
toggleNodeDisabledState(key, state) {
const node = this._getNodeByKey(key);
if (!node) {
return
}
const items = this._filesTreeView.option("items");
const itemIndex = items.map((item => item.getInternalKey())).indexOf(node.getInternalKey());
if (-1 !== itemIndex) {
this._filesTreeView.option(`items[${itemIndex}].disabled`, state)
}
}
_saveScrollTopPosition() {
if (!(0, _window.hasWindow)()) {
return
}
this._scrollTopPosition = this._filesTreeView.getScrollable().scrollTop()
}
_restoreScrollTopPosition() {
if (!(0, _window.hasWindow)() || !(0, _type.isNumeric)(this._scrollTopPosition)) {
return
}
setTimeout((() => this._filesTreeView.getScrollable().scrollTo(this._scrollTopPosition)))
}
_updateFocusedElement() {
const directoryInfo = this._getCurrentDirectory();
const $element = this._getItemElementByKey(null === directoryInfo || void 0 === directoryInfo ? void 0 : directoryInfo.getInternalKey());
if (this._$focusedElement) {
this._$focusedElement.toggleClass("dx-filemanager-focused-item", false)
}
this._$focusedElement = $element || (0, _renderer.default)();
this._$focusedElement.toggleClass("dx-filemanager-focused-item", true)
}
_getNodeByKey(key) {
var _this$_filesTreeView;
return null === (_this$_filesTreeView = this._filesTreeView) || void 0 === _this$_filesTreeView ? void 0 : _this$_filesTreeView._getNode(key)
}
_getPublicNode(key) {
var _this$_filesTreeView2;
const nodesQueue = [...null === (_this$_filesTreeView2 = this._filesTreeView) || void 0 === _this$_filesTreeView2 ? void 0 : _this$_filesTreeView2.getNodes()];
while (nodesQueue.length) {
const node = nodesQueue.shift();
if (node.itemData.getInternalKey() === key) {
return node
} else if (node.children.length) {
nodesQueue.push(...node.children)
}
}
return
}
_getItemElementByKey(key) {
const node = this._getNodeByKey(key);
if (node) {
const $node = this._filesTreeView._getNodeElement(node);
if ($node) {
return $node.children(this._filesTreeViewItemSelector)
}
}
return null
}
_getDefaultOptions() {
return (0, _extend.extend)(super._getDefaultOptions(), {
storeExpandedState: false,
initialFolder: null,
contextMenu: null,
getItems: null,
getCurrentDirectory: null,
onDirectoryClick: null
})
}
_optionChanged(args) {
const name = args.name;
switch (name) {
case "storeExpandedState":
this._storeExpandedState = this.option(name);
break;
case "getItems":
case "rootFolderDisplayName":
case "initialFolder":
case "contextMenu":
this.repaint();
break;
case "getCurrentDirectory":
this.getCurrentDirectory = this.option(name);
break;
case "onDirectoryClick":
case "onFilesTreeViewContentReady":
this._actions[name] = this._createActionByOption(name);
break;
default:
super._optionChanged(args)
}
}
get _filesTreeViewItemSelector() {
return ".dx-treeview-item"
}
get _contextMenu() {
return this.option("contextMenu")
}
toggleDirectoryExpandedState(directoryInfo, state) {
const deferred = new _deferred.Deferred;
const treeViewNode = this._getPublicNode(null === directoryInfo || void 0 === directoryInfo ? void 0 : directoryInfo.getInternalKey());
if (!treeViewNode) {
return deferred.reject().promise()
}
if (treeViewNode.expanded === state || treeViewNode.itemsLoaded && !treeViewNode.itemData.fileItem.hasSubDirectories) {
return deferred.resolve().promise()
}
const action = state ? "expandItem" : "collapseItem";
return this._filesTreeView[action](directoryInfo.getInternalKey())
}
refresh() {
this._$focusedElement = null;
this._saveScrollTopPosition();
this._filesTreeView.option("dataSource", [])
}
updateCurrentDirectory() {
if (this._disposed) {
return
}
this._updateFocusedElement();
this._storeExpandedState && this._updateExpandedStateToCurrentDirectory()
}
_updateExpandedStateToCurrentDirectory() {
return this.toggleDirectoryExpandedStateRecursive(this._getCurrentDirectory().parentDirectory, true)
}
toggleDirectoryExpandedStateRecursive(directoryInfo, state) {
const dirLine = [];
for (let dirInfo = directoryInfo; dirInfo; dirInfo = dirInfo.parentDirectory) {
dirLine.unshift(dirInfo)
}
return this.toggleDirectoryLineExpandedState(dirLine, state)
}
toggleDirectoryLineExpandedState(dirLine, state) {
if (!dirLine.length) {
return (new _deferred.Deferred).resolve().promise()
}
return this.toggleDirectoryExpandedState(dirLine.shift(), state).then((() => this.toggleDirectoryLineExpandedState(dirLine, state)))
}
}
var _default = exports.default = FileManagerFilesTreeView;
module.exports = exports.default;
module.exports.default = exports.default;