diff options
| author | Shulhan <ms@kilabit.info> | 2021-08-29 13:57:33 +0700 |
|---|---|---|
| committer | Shulhan <ms@kilabit.info> | 2021-08-29 13:57:33 +0700 |
| commit | 1364ab29de697993bc34832cf77d4e99ee037223 (patch) | |
| tree | b8dcb3e89020c736d02a04277bad0a16933e5224 | |
| parent | 7940feda7d50c404fb245b645c8261d6cb4dd768 (diff) | |
| download | pakakeh.ts-1364ab29de697993bc34832cf77d4e99ee037223.tar.xz | |
vfs: simplify the handlers on WuiVfsOptions
Previously, the WuiVfsOptions contains two handlers: ListNodes to
retrieve list of files and OnClickNode for triggering changes when
user click a path or item in the list.
This changes simplify it into single handler: Open, which accept
path and boolean is_dir.
| -rw-r--r-- | vfs/example.html | 36 | ||||
| -rw-r--r-- | vfs/vfs.d.ts | 22 | ||||
| -rw-r--r-- | vfs/vfs.js | 190 | ||||
| -rw-r--r-- | vfs/vfs.ts | 180 |
4 files changed, 153 insertions, 275 deletions
diff --git a/vfs/example.html b/vfs/example.html index 3d8a772..1ebc1b4 100644 --- a/vfs/example.html +++ b/vfs/example.html @@ -9,7 +9,7 @@ <script> var exports = {} - var wui_vfs; + var wui_vfs </script> <script src="vfs.js"></script> <script> @@ -86,38 +86,28 @@ async function main() { let opts = { id: "vfs", - ListNodes: doListNodes, - OnClickNode: OnClickNode, + Open: Open, } - let res = await NewWuiVfs(opts) - if (res.code != 200) { - console.error(res) - return - } - wui_vfs = res.data + wui_vfs = new WuiVfs(opts) + wui_vfs.OpenDir("/") } - function doListNodes() { + function Open(path, is_dir) { + console.log("Open:", path, is_dir) let res = { code: 200, - data: dummyfs, } - return res - } - function OnClickNode(path, is_dir) { if (is_dir) { - return + res.data = dummyfs[path] + return res } - let res = { - code: 200, - data: { - name: "", - path: path, - content: "", - }, + res.data = { + name: "", + path: path, + content: "", } switch (path) { @@ -142,7 +132,7 @@ res.message = "path not found" } - console.log(res) + console.log("Open:", res) } </script> </body> diff --git a/vfs/vfs.d.ts b/vfs/vfs.d.ts index 706ccc8..9a9ea22 100644 --- a/vfs/vfs.d.ts +++ b/vfs/vfs.d.ts @@ -12,18 +12,26 @@ export interface WuiVfsNodeInterface { } export interface WuiVfsOptions { id: string; - ListNodes: () => WuiResponseInterface; - OnClickNode(path: string, is_dir: boolean): void; + Open(path: string, is_dir: boolean): WuiResponseInterface; } -export declare function NewWuiVfs(opts: WuiVfsOptions): Promise<WuiResponseInterface>; export declare class WuiVfs { opts: WuiVfsOptions; private el; private com_path; private com_list; - private path_node; constructor(opts: WuiVfsOptions); - init(): Promise<WuiResponseInterface>; - private open; - OpenPath(this: WuiVfs, path: string): void; + OnClickNode(node: WuiVfsNode): void; + OpenDir(path: string): Promise<void>; } +declare class WuiVfsNode implements WuiVfsNodeInterface { + path: string; + name: string; + mod_time_epoch: number; + mod_time_rfc3339: string; + size: number; + mode: string; + is_dir: boolean; + childs: WuiVfsNode[]; + constructor(opts: WuiVfsNodeInterface); +} +export {}; @@ -39,105 +39,62 @@ var __generator = (this && this.__generator) || function (thisArg, body) { } }; exports.__esModule = true; -exports.WuiVfs = exports.NewWuiVfs = void 0; -function NewWuiVfs(opts) { - return __awaiter(this, void 0, void 0, function () { - var vfs, res; - return __generator(this, function (_a) { - switch (_a.label) { - case 0: - vfs = new WuiVfs(opts); - return [4 /*yield*/, vfs.init()]; - case 1: - res = _a.sent(); - if (res.code != 200) { - return [2 /*return*/, res]; - } - res.data = vfs; - return [2 /*return*/, res]; - } - }); - }); -} -exports.NewWuiVfs = NewWuiVfs; +exports.WuiVfs = void 0; var WuiVfs = /** @class */ (function () { function WuiVfs(opts) { + var _this = this; + this.opts = opts; this.opts = opts; - this.path_node = {}; var el = document.getElementById(opts.id); if (!el) { console.error("WuiVfs: element id", opts.id, "not found"); return; } this.el = el; + this.com_path = new WuiVfsPath(function (path) { + _this.OpenDir(path); + }); + this.el.appendChild(this.com_path.el); + this.com_list = new WuiVfsList(function (node) { + _this.OnClickNode(node); + }); + this.el.appendChild(this.com_list.el); } - // - // (0) Fetch the list of nodes from remote server and store it in path_node. - // (1) Create the WuiVfsPath - // (2) Create the WuiVfsList - // (3) Open the root "/" - // - WuiVfs.prototype.init = function () { + // OnClickNode is a handler that will be called when a node is clicked + // inside the WuiVfsList. + WuiVfs.prototype.OnClickNode = function (node) { + if (!node.is_dir) { + this.opts.Open(node.path, false); + return; + } + this.OpenDir(node.path); + }; + // OpenDir is a handler that will be called when a path is clicked + // inside the WuiVfsPath. + WuiVfs.prototype.OpenDir = function (path) { return __awaiter(this, void 0, void 0, function () { - var res, res_path_node, key, value, node; - var _this = this; + var res, node; return __generator(this, function (_a) { switch (_a.label) { - case 0: return [4 /*yield*/, this.opts.ListNodes()]; + case 0: return [4 /*yield*/, this.opts.Open(path, true)]; case 1: res = _a.sent(); if (res.code != 200) { - return [2 /*return*/, res]; - } - res_path_node = res.data; - for (key in res_path_node) { - value = res_path_node[key]; - node = new WuiVfsNode(value, function (node) { - if (_this.opts.OnClickNode) { - _this.opts.OnClickNode(node.path, node.is_dir); - } - if (node.is_dir) { - _this.open(node); - } - }); - this.path_node[key] = node; + return [2 /*return*/]; } - this.el.innerHTML = ""; - // (1) - this.com_path = new WuiVfsPath(function (path) { - _this.OpenPath(path); - }); - this.el.appendChild(this.com_path.el); - // (2) - this.com_list = new WuiVfsList(); - this.el.appendChild(this.com_list.el); - // (3) - this.open(this.path_node["/"]); - return [2 /*return*/, res]; + node = new WuiVfsNode(res.data); + this.com_path.Open(node); + this.com_list.Open(node); + return [2 /*return*/]; } }); }); }; - WuiVfs.prototype.open = function (node) { - this.com_path.Open(node); - this.com_list.Open(node); - }; - // OpenPath is a handler that will be called when the directory name on - // top of UI clicked. - WuiVfs.prototype.OpenPath = function (path) { - var node = this.path_node[path]; - if (!node) { - console.error("WuiVfs: OpenPath: invalid path: ", path); - return; - } - this.open(node); - }; return WuiVfs; }()); exports.WuiVfs = WuiVfs; var WuiVfsNode = /** @class */ (function () { - function WuiVfsNode(opts, onClick) { - var _this = this; + function WuiVfsNode(opts) { this.path = opts.path || ""; this.name = opts.name || ""; this.mod_time_epoch = opts.mod_time_epoch || 0; @@ -146,57 +103,54 @@ var WuiVfsNode = /** @class */ (function () { this.mode = opts.mode || ""; this.is_dir = opts.is_dir || false; this.childs = []; - if (opts.childs !== undefined) { + if (opts.childs) { for (var _i = 0, _a = opts.childs; _i < _a.length; _i++) { var c = _a[_i]; - this.childs.push(new WuiVfsNode(c, onClick)); + this.childs.push(new WuiVfsNode(c)); } } - this.el = document.createElement("div"); - this.el.style.padding = "1em"; - this.el.style.cursor = "pointer"; - this.el.innerHTML = this.name; - if (this.is_dir) { - this.el.style.backgroundColor = "cornsilk"; - } - this.el.onclick = function (event) { - onClick(_this); - }; - this.el.onmouseout = function (event) { - _this.onMouseOut(_this); - }; - this.el.onmouseover = function (event) { - _this.onMouseOver(_this); - }; } - WuiVfsNode.prototype.onMouseOut = function (t) { - if (this.is_dir) { - this.el.style.backgroundColor = "cornsilk"; - } - else { - t.el.style.backgroundColor = "white"; - } - }; - WuiVfsNode.prototype.onMouseOver = function (t) { - t.el.style.backgroundColor = "aliceblue"; - }; return WuiVfsNode; }()); var WuiVfsList = /** @class */ (function () { - function WuiVfsList() { + function WuiVfsList(onClick) { + this.onClick = onClick; this.el = document.createElement("div"); this.el.style.borderWidth = "1px"; this.el.style.borderStyle = "solid"; this.el.style.borderColor = "silver"; } WuiVfsList.prototype.Open = function (node) { + var _this = this; this.el.innerHTML = ""; - if (node.childs === undefined) { - return; - } + var _loop_1 = function (c) { + var el = document.createElement("div"); + el.style.padding = "1em"; + el.style.cursor = "pointer"; + el.innerHTML = c.name; + if (c.is_dir) { + el.style.backgroundColor = "cornsilk"; + } + el.onclick = function (ev) { + _this.onClick(c); + }; + el.onmouseout = function (event) { + if (c.is_dir) { + el.style.backgroundColor = "cornsilk"; + } + else { + el.style.backgroundColor = "white"; + } + }; + el.onmouseover = function (event) { + el.style.backgroundColor = "aliceblue"; + }; + this_1.el.appendChild(el); + }; + var this_1 = this; for (var _i = 0, _a = node.childs; _i < _a.length; _i++) { var c = _a[_i]; - this.el.appendChild(c.el); + _loop_1(c); } }; return WuiVfsList; @@ -221,7 +175,7 @@ var WuiVfsPath = /** @class */ (function () { else { paths = node.path.split("/"); } - var _loop_1 = function (x) { + var _loop_2 = function (x) { var full_path = ""; var p = ""; if (x == 0) { @@ -241,24 +195,18 @@ var WuiVfsPath = /** @class */ (function () { _this.onClick(full_path); }; crumb.onmouseout = function (event) { - _this.onMouseOut(crumb, event); + crumb.style.backgroundColor = "white"; }; crumb.onmouseover = function (event) { - _this.onMouseOver(crumb, event); + crumb.style.backgroundColor = "aliceblue"; }; - this_1.el.appendChild(crumb); + this_2.el.appendChild(crumb); }; - var this_1 = this; + var this_2 = this; for (var x = 0; x < paths.length; x++) { - _loop_1(x); + _loop_2(x); } }; - WuiVfsPath.prototype.onMouseOut = function (crumb, event) { - crumb.style.backgroundColor = "white"; - }; - WuiVfsPath.prototype.onMouseOver = function (crumb, event) { - crumb.style.backgroundColor = "aliceblue"; - }; return WuiVfsPath; }()); -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vfs.js","sourceRoot":"","sources":["vfs.ts"],"names":[],"mappings":"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"}
\ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmZzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsidmZzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSxrRUFBa0U7QUFDbEUseUVBQXlFO0FBQ3pFLDZCQUE2Qjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBeUI3QjtJQUtDLGdCQUFtQixJQUFtQjtRQUF0QyxpQkFtQkM7UUFuQmtCLFNBQUksR0FBSixJQUFJLENBQWU7UUFDckMsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUE7UUFFaEIsSUFBSSxFQUFFLEdBQUcsUUFBUSxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUE7UUFDekMsSUFBSSxDQUFDLEVBQUUsRUFBRTtZQUNSLE9BQU8sQ0FBQyxLQUFLLENBQUMsb0JBQW9CLEVBQUUsSUFBSSxDQUFDLEVBQUUsRUFBRSxXQUFXLENBQUMsQ0FBQTtZQUN6RCxPQUFNO1NBQ047UUFDRCxJQUFJLENBQUMsRUFBRSxHQUFHLEVBQUUsQ0FBQTtRQUVaLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxVQUFVLENBQUMsVUFBQyxJQUFZO1lBQzNDLEtBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUE7UUFDbkIsQ0FBQyxDQUFDLENBQUE7UUFDRixJQUFJLENBQUMsRUFBRSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFBO1FBRXJDLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxVQUFVLENBQUMsVUFBQyxJQUFnQjtZQUMvQyxLQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFBO1FBQ3ZCLENBQUMsQ0FBQyxDQUFBO1FBQ0YsSUFBSSxDQUFDLEVBQUUsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUMsQ0FBQTtJQUN0QyxDQUFDO0lBRUQsc0VBQXNFO0lBQ3RFLHlCQUF5QjtJQUN6Qiw0QkFBVyxHQUFY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
\ No newline at end of file @@ -20,97 +20,57 @@ export interface WuiVfsNodeInterface { export interface WuiVfsOptions { id: string - ListNodes: () => WuiResponseInterface - - // OnClickNode define an handler that will be called everytime a node is - // clicked. The is_dir will be true, if the node is a directory. - OnClickNode(path: string, is_dir: boolean): void -} - -export async function NewWuiVfs(opts: WuiVfsOptions): Promise<WuiResponseInterface> { - let vfs = new WuiVfs(opts) - let res = await vfs.init() - if (res.code != 200) { - return res - } - res.data = vfs - return res + // Open define an handler that will be called when a directory or a file + // is clicked from the WuiVfsPath or WuiVfsList. + Open(path: string, is_dir: boolean): WuiResponseInterface } export class WuiVfs { private el!: HTMLElement private com_path!: WuiVfsPath private com_list!: WuiVfsList - private path_node: WuiPathNode = {} constructor(public opts: WuiVfsOptions) { + this.opts = opts + let el = document.getElementById(opts.id) if (!el) { console.error("WuiVfs: element id", opts.id, "not found") return } this.el = el - } - - // - // (0) Fetch the list of nodes from remote server and store it in path_node. - // (1) Create the WuiVfsPath - // (2) Create the WuiVfsList - // (3) Open the root "/" - // - async init(): Promise<WuiResponseInterface> { - // (0) - let res = await this.opts.ListNodes() - if (res.code != 200) { - return res - } - - let res_path_node = res.data as WuiPathNodeInterface - for (const key in res_path_node) { - const value = res_path_node[key] as WuiVfsNodeInterface - const node = new WuiVfsNode(value, (node: WuiVfsNode) => { - if (this.opts.OnClickNode) { - this.opts.OnClickNode(node.path, node.is_dir) - } - if (node.is_dir) { - this.open(node) - } - }) - this.path_node[key] = node - } - - this.el.innerHTML = "" - // (1) this.com_path = new WuiVfsPath((path: string) => { - this.OpenPath(path) + this.OpenDir(path) }) this.el.appendChild(this.com_path.el) - // (2) - this.com_list = new WuiVfsList() + this.com_list = new WuiVfsList((node: WuiVfsNode) => { + this.OnClickNode(node) + }) this.el.appendChild(this.com_list.el) - - // (3) - this.open(this.path_node["/"]) - - return res } - private open(node: WuiVfsNode) { - this.com_path.Open(node) - this.com_list.Open(node) + // OnClickNode is a handler that will be called when a node is clicked + // inside the WuiVfsList. + OnClickNode(node: WuiVfsNode): void { + if (!node.is_dir) { + this.opts.Open(node.path, false) + return + } + this.OpenDir(node.path) } - // OpenPath is a handler that will be called when the directory name on - // top of UI clicked. - OpenPath(this: WuiVfs, path: string) { - const node = this.path_node[path] - if (!node) { - console.error("WuiVfs: OpenPath: invalid path: ", path) + // OpenDir is a handler that will be called when a path is clicked + // inside the WuiVfsPath. + async OpenDir(path: string): Promise<void> { + let res = await this.opts.Open(path, true) + if (res.code != 200) { return } - this.open(node) + let node = new WuiVfsNode(res.data as WuiVfsNodeInterface) + this.com_path.Open(node) + this.com_list.Open(node) } } @@ -124,9 +84,7 @@ class WuiVfsNode implements WuiVfsNodeInterface { is_dir: boolean childs: WuiVfsNode[] - el: HTMLElement - - constructor(opts: WuiVfsNodeInterface, onClick: NodeClickHandler) { + constructor(opts: WuiVfsNodeInterface) { this.path = opts.path || "" this.name = opts.name || "" this.mod_time_epoch = opts.mod_time_epoch || 0 @@ -136,48 +94,18 @@ class WuiVfsNode implements WuiVfsNodeInterface { this.is_dir = opts.is_dir || false this.childs = [] - if (opts.childs !== undefined) { + if (opts.childs) { for (let c of opts.childs) { - this.childs.push(new WuiVfsNode(c, onClick)) + this.childs.push(new WuiVfsNode(c)) } } - - this.el = document.createElement("div") - this.el.style.padding = "1em" - this.el.style.cursor = "pointer" - this.el.innerHTML = this.name - - if (this.is_dir) { - this.el.style.backgroundColor = "cornsilk" - } - - this.el.onclick = (event) => { - onClick(this) - } - this.el.onmouseout = (event) => { - this.onMouseOut(this) - } - this.el.onmouseover = (event) => { - this.onMouseOver(this) - } - } - - onMouseOut(t: WuiVfsNode) { - if (this.is_dir) { - this.el.style.backgroundColor = "cornsilk" - } else { - t.el.style.backgroundColor = "white" - } - } - onMouseOver(t: WuiVfsNode) { - t.el.style.backgroundColor = "aliceblue" } } class WuiVfsList { el: HTMLElement - constructor() { + constructor(public onClick: NodeClickHandler) { this.el = document.createElement("div") this.el.style.borderWidth = "1px" this.el.style.borderStyle = "solid" @@ -187,12 +115,31 @@ class WuiVfsList { Open(node: WuiVfsNode) { this.el.innerHTML = "" - if (node.childs === undefined) { - return - } - for (let c of node.childs) { - this.el.appendChild(c.el) + let el = document.createElement("div") + el.style.padding = "1em" + el.style.cursor = "pointer" + el.innerHTML = c.name + + if (c.is_dir) { + el.style.backgroundColor = "cornsilk" + } + + el.onclick = (ev: MouseEvent) => { + this.onClick(c) + } + el.onmouseout = (event) => { + if (c.is_dir) { + el.style.backgroundColor = "cornsilk" + } else { + el.style.backgroundColor = "white" + } + } + el.onmouseover = (event) => { + el.style.backgroundColor = "aliceblue" + } + + this.el.appendChild(el) } } } @@ -244,31 +191,16 @@ class WuiVfsPath { this.onClick(full_path) } crumb.onmouseout = (event) => { - this.onMouseOut(crumb, event) + crumb.style.backgroundColor = "white" } crumb.onmouseover = (event) => { - this.onMouseOver(crumb, event) + crumb.style.backgroundColor = "aliceblue" } this.el.appendChild(crumb) } } - - private onMouseOut(crumb: HTMLElement, event: MouseEvent) { - crumb.style.backgroundColor = "white" - } - private onMouseOver(crumb: HTMLElement, event: MouseEvent) { - crumb.style.backgroundColor = "aliceblue" - } -} - -type WuiPathNode = { - [key: string]: WuiVfsNode } -type WuiPathNodeInterface = { - [key: string]: WuiVfsNodeInterface -} - -type NodeClickHandler = (node: WuiVfsNode) => void type PathClickHandler = (path: string) => void +type NodeClickHandler = (node: WuiVfsNode) => void |
