From 7cfca1b0f23fac33063bf6e8bfc18762013fcd18 Mon Sep 17 00:00:00 2001 From: "shylinux@163.com" Date: Mon, 19 Sep 2022 23:07:13 +0800 Subject: [PATCH] opt vimer --- frame.js | 10 +- lib/misc.js | 5 + lib/page.js | 8 +- page/index.css | 7 +- panel/footer.js | 2 +- panel/header.js | 2 +- panel/search.js | 9 +- plugin/local/code/inner.css | 16 +- plugin/local/code/inner.js | 879 ++++++++++++++---------------- plugin/local/code/inner/syntax.js | 19 + plugin/local/code/vimer.js | 253 ++++----- plugin/local/code/xterm.js | 4 +- plugin/local/team/plan.js | 2 +- plugin/local/wiki/draw.js | 4 +- plugin/local/wiki/word.js | 4 +- plugin/state.js | 5 +- plugin/story/parse.js | 2 +- plugin/table.js | 38 +- proto.js | 5 + 19 files changed, 620 insertions(+), 654 deletions(-) diff --git a/frame.js b/frame.js index 0c41e971..e278d703 100644 --- a/frame.js +++ b/frame.js @@ -372,7 +372,7 @@ Volcanos(chat.ONAPPEND, {help: "渲染引擎", _init: function(can, meta, list, } return {text: [value, html.TD], onclick: function(event) { var target = event.target - if (can.page.tagis(html.INPUT, target) && target.type == html.BUTTON) { return run([ctx.ACTION, target.name]) } + if (can.page.tagis(target, html.INPUT) && target.type == html.BUTTON) { return run([ctx.ACTION, target.name]) } if (key == mdb.HASH && can.user.mod.isDiv) { return can.user.jumps("/chat/div/"+value) } if (can.sup.onaction.change(event, can.sup, key, event.target.innerText).length == 0) { can.sup && can.sup._item_click && can.sup._item_click(value, key) @@ -581,7 +581,7 @@ Volcanos(chat.ONMOTION, {help: "动态特效", _init: function(can, target) { var list = can.core.List(arguments).slice(2) if (list.length == 0) { list = [chat.CARTE, chat.INPUT] } can.page.Modify(can, target, {onmouseover: function(event) { - if (can.page.tagis(html.IMG, event.target)) { return } + if (can.page.tagis(event.target, html.IMG)) { return } can.core.List(list, function(key, index) { that.del(can, key) }) }}) }, @@ -812,7 +812,7 @@ Volcanos(chat.ONKEYMAP, {help: "键盘交互", _focus: [], _init: function(can, if (window.webview) { if (event.target.tagName == "A") { can.user.open(event.target.href) } } - if (can.page.tagis([html.SELECT, html.INPUT, html.TEXTAREA], event.target)) { return } + if (can.page.tagis( event.target, html.SELECT, html.INPUT, html.TEXTAREA)) { return } can.page.Select(can, document.body, can.page.Keys("fieldset.input.key.float"), function(item) { can.page.Remove(can, item) }) @@ -829,7 +829,7 @@ Volcanos(chat.ONKEYMAP, {help: "键盘交互", _focus: [], _init: function(can, } } return } - if (can.page.tagis([html.SELECT, html.INPUT, html.TEXTAREA], event.target)) { return } + if (can.page.tagis(event.target, html.SELECT, html.INPUT, html.TEXTAREA)) { return } var msg = can.request(event, {"model": "normal"}); if (msg.Option(ice.MSG_HANDLE) == ice.TRUE) { return } can.onengine.signal(can, chat.ONKEYDOWN, msg); if (msg.Option(ice.MSG_HANDLE) == ice.TRUE) { return } can._keylist = can.onkeymap._parse(event, can, msg.Option("model"), can._keylist, can._output) @@ -869,7 +869,7 @@ Volcanos(chat.ONKEYMAP, {help: "键盘交互", _focus: [], _init: function(can, Escape: function(event, can, target) { target.blur() }, Enter: function(event, can, target) { var his = target._history||[] his.push(target.value), target._history = his, target._current = his.length - can.page.tagis(html.INPUT, event.target) && can.onmotion.focus(can, target) + can.page.tagis(event.target, html.INPUT) && can.onmotion.focus(can, target) }, }, insert_ctrl: { diff --git a/lib/misc.js b/lib/misc.js index 9d52c92a..6da25b0f 100644 --- a/lib/misc.js +++ b/lib/misc.js @@ -232,6 +232,11 @@ Volcanos("misc", {help: "通信协议", Message: function(event, can) { var msg var search = can.core.Item(args, function(key, value) { return key+"="+encodeURIComponent(value) }).join("&") return search? location.search = search: location.href = location.pathname }), + MergePodCmd: shy("地址链接", function(can, objs) { + objs.pod = can.core.Keys(can.misc.Search(can, "pod"), objs.pod) + objs.topic = can.misc.Search(can, "topic") + return can.misc.MergeURL(can, objs, true) + }), MergeURL: shy("地址链接", function(can, objs, clear) { var path = location.pathname; objs._path && (path = objs._path), delete(objs._path) objs.pod && (path = "/chat/pod/"+objs.pod), delete(objs.pod) diff --git a/lib/page.js b/lib/page.js index 895dc1d2..89e2f346 100644 --- a/lib/page.js +++ b/lib/page.js @@ -309,6 +309,7 @@ Volcanos("page", {help: "用户界面", ClassList: { switch (type) { case html.A: return ""+(args[2]||args[1])+"" case html.IMG: return args[2]? "": "" + case html.SPAN: return args[2]? ""+args[1]+"": args[1] } }, replace: function(can, text, key, value) { @@ -443,8 +444,11 @@ Volcanos("page", {help: "用户界面", ClassList: { styleSheet.type = "text/css", styleSheet.innerText = text document.head.appendChild(styleSheet) }, - tagis: function(type, target) { type = typeof type == lang.OBJECT? type: [type] - if (type.indexOf(target.tagName.toLowerCase()) > -1) { return true } + tagis: function(target) { + var type = target.tagName.toLowerCase() + for (var i = 1; i < arguments.length; i++) { + if (type == arguments[i]) { return true } + } }, offsetTop: function(item) { var res = 0 while (item) { res += item.offsetTop||0, item = item.parentNode } diff --git a/page/index.css b/page/index.css index 30c0b694..ca008fa6 100644 --- a/page/index.css +++ b/page/index.css @@ -46,7 +46,7 @@ div.output div.project div.item { padding:2px 10px; } div.output div.project div.list { margin-left:10px; } div.output div.project div.switch { width:12px; float:left; transform: rotate(90deg) translate(1px, 0px); } div.output div.project div.switch.open { transform: rotate(180deg) translate(-4px, 4px); } -div.output div.project div.zone>div.name { background-color:steelblue; color:white; text-align:center; padding:5px; clear:both; } +div.output div.project div.zone>div.name { background-color:steelblue; color:white; text-align:center; padding:3px; clear:both; } div.output div.project div.item>div.name { padding-left:20px; } div.output div.project div.zone>div.action>div.item { float:right; clear:none; padding:0; margin:0; } div.output div.project div.zone>div.action>div.item input[type=text] { background-color:#ff000000; padding-left:10px; color:white; } @@ -59,10 +59,8 @@ fieldset.story>legend { margin:10px 0; } fieldset.plug>legend { float:left; margin-right:5px; } fieldset.float>legend { float:left; margin-right:5px; } fieldset.full>legend { float:left; margin:0; margin-right:5px; } -fieldset.float input[type=button][name=close]{ display:block; } -fieldset.plug>form.option input[type=button][name=close]{ display:block; } -fieldset.full input[type=button][name=close]{ display:block; } fieldset.plugin>form.option input[type=button][name=close]{ display:none; } +fieldset.word>div.output>fieldset.story>form.option input[type=button][name=close]{ display:none; } fieldset.plugin div.status { border-top:1px solid darkcyan; } fieldset.story div.status { border-top:1px solid darkcyan; } fieldset.output>form.option { display:none; } @@ -315,6 +313,7 @@ body.simple fieldset.feel.float { top:0; } div.output div.row div.col { float:left; border:solid 1px green; } body.white.simple div.profile { background-color:whitesmoke; } +body.white.simple div.profile { color:black; } body.white.simple div.display { background-color:whitesmoke; } body.white.simple div.code { background-color:white; color:black; border:none; } body.white.simple fieldset.plugin div.status { border-top:1px solid lightgray; } diff --git a/panel/footer.js b/panel/footer.js index 675fa9af..c02f0469 100644 --- a/panel/footer.js +++ b/panel/footer.js @@ -72,7 +72,7 @@ Volcanos(chat.ONEXPORT, {help: "导出数据", can.onappend._action(can, [cli.CLOSE, cli.REFRESH, {input: html.TEXT, placeholder: "filter", _init: function(input) { can.onengine.signal(can, "keymap.focus", can.request({}, {cb: function(event) { - if (can.page.tagis(html.INPUT, event.target)) { return } + if (can.page.tagis(event.target, html.INPUT)) { return } if (event.key == lang.ESCAPE) { ui.close(); return } if (event.key == ice.SP) { input.focus(), can.onkeymap.prevent(event) } }})) diff --git a/panel/header.js b/panel/header.js index f9498660..650522d3 100644 --- a/panel/header.js +++ b/panel/header.js @@ -41,7 +41,7 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar _search: function(can, msg, target) { var ui = can.onappend.input(can, {type: html.TEXT, name: mdb.SEARCH, onkeydown: function(event) { can.onkeymap.input(event, can); switch (event.key) { - case lang.ENTER: can.onengine.signal(can, chat.ONOPENSEARCH, can.request(event, {type: "command,space,text"||mdb.FOREACH, word: event.target.value})) + case lang.ENTER: can.onengine.signal(can, chat.ONOPENSEARCH, can.request(event, {type: "command,space,word,text"||mdb.FOREACH, word: event.target.value})) } }}, "", target, "title search") can.onimport.menu(can, mdb.SEARCH, function() { diff --git a/panel/search.js b/panel/search.js index dbbc49ac..cc2acdfb 100644 --- a/panel/search.js +++ b/panel/search.js @@ -1,6 +1,6 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, list, cb, target) { can._foreach = "*" - can._foreach = "command,space,text" + can._foreach = "command,space,word,text" can.list = msg.Table(), can.onmotion.clear(can, can.ui.content) var table = can.onappend.table(can, msg, function(value, key, index, line, array) { can.Status(mdb.TOTAL, index+1) return {text: [key == mdb.TEXT && can.base.isFunc(line.text) && line.text.help || value, html.TD], onclick: function(event) { @@ -109,13 +109,12 @@ Volcanos(chat.ONACTION, {help: "交互操作", list: [cli.CLOSE, cli.CLEAR, cli. }}]), can.Status("selected", can.page.Select(can, can.ui.display, html.TR).length-1) }, - plugin: function(event, can, index) { var line = can.list[index] + plugin: function(event, can, index) { var line = can.list[index], args = [] if (can.base.isFunc(line.text)) { return can.onmotion.hide(can), line.text(event) } - if (line.ctx == "web.wiki" && line.cmd == "word") { return } - if (can.page.tagis(html.A, event.target)) { return } + if (can.page.tagis(event.target, html.A)) { return } var cmd = line.cmd == ctx.COMMAND? can.core.Keys(line.type, line.name.split(ice.SP)[0]): can.core.Keys(line.ctx, line.cmd) - can.onappend.plugin(can, {type: "plug", index: cmd||msg.Option(mdb.INDEX)}, function(sub, meta) { + can.onappend.plugin(can, {type: "story", index: cmd||msg.Option(mdb.INDEX), args: cmd == "web.wiki.word"? [line.name]: args}, function(sub, meta) { sub.run = function(event, cmds, cb) { can.runActionCommand(event, meta.index, cmds, cb) } can.page.style(can, sub._output, html.MAX_WIDTH, sub.ConfWidth(can.ConfWidth())) sub.ConfHeight(can.ConfHeight()-2*html.ACTION_HEIGHT-117) diff --git a/plugin/local/code/inner.css b/plugin/local/code/inner.css index 459d5335..340b8d0b 100644 --- a/plugin/local/code/inner.css +++ b/plugin/local/code/inner.css @@ -30,9 +30,12 @@ fieldset.inner>div.output td.profile iframe { border:0; } fieldset.inner>div.output td.content>div.tabs { display:none; } fieldset.inner>div.output td.content>div.path { display:none; } -fieldset.inner div.output fieldset.toolkit { position:absolute; bottom:0px; right:0px; } +fieldset.inner div.output fieldset.toolkit { position:relative; bottom:0px; right:0px; } fieldset.inner>div.output fieldset.toolkit>div.output>fieldset { display:none; } fieldset.inner>div.output fieldset.toolkit>div.output>fieldset.select { background-color:#0e3369b3; color:white; display:block; z-index:10; } +fieldset.inner>div.output fieldset.toolkit>div.output fieldset { display:none; } +fieldset.inner>div.output fieldset.toolkit>div.output fieldset.select { background-color:#0e3369b3; color:white; display:block; z-index:10; } +fieldset.inner div.output fieldset.toolkit>div.output { position:unset; } fieldset>div.output>fieldset.plug { position:absolute; bottom:0px; right:0px; } fieldset>div.output>fieldset.plug { display:none; } @@ -40,7 +43,12 @@ fieldset>div.output>fieldset.plug.select { background-color:#0e3369b3; color:whi fieldset>div.status legend { float:right; height:30px; } fieldset>div.status legend.select { background-color:green; } fieldset>div.status legend:hover { background-color:green; } -fieldset>div.status { height:31px; overflow:auto; } +fieldset.inner>div.status { height:31px; overflow:auto; } +fieldset.panel>div.status { display:none; overflow:auto; } +fieldset.inner.simple>legend { display:none; } +fieldset.inner.simple>form.option { display:none; } +fieldset.inner.simple>div.status { display:none; } +fieldset.inner.simple div.toggle { display:none; } fieldset.Action>div.status { display:none; } @@ -50,8 +58,8 @@ body.black fieldset.inner.float>div.output div.content { color:white; } /* body.white fieldset.inner.float>div.output div.profile { color:black; } */ -body.simple fieldset.inner>div.output td.content>div.tabs { background-color:gray; padding:0px; height:31px; overflow:auto; display:block; } -body.simple fieldset.inner>div.output td.content>div.tabs div.tabs { background-color:gray; font-family:monospace; padding:7px 20px; height:17px; float:left; } +body.simple fieldset.inner>div.output td.content>div.tabs { background-color:slategrey; padding:0px; height:31px; overflow:auto; display:block; } +body.simple fieldset.inner>div.output td.content>div.tabs div.tabs { background-color:teal; font-family:monospace; padding:7px 20px; height:17px; float:left; } body.simple fieldset.inner>div.output td.content>div.tabs div.tabs.select { background-color:#1d3349; } body.simple fieldset.inner>div.output td.content>div.tabs div.tabs:hover { background-color:#1d3349; } body.simple fieldset.inner>div.output td.content>div.path { background-color:#1d3349; padding:4px 10px; margin-bottom:4px; display:block; clear:both; } diff --git a/plugin/local/code/inner.js b/plugin/local/code/inner.js index 81e23942..a1f9ae7e 100644 --- a/plugin/local/code/inner.js +++ b/plugin/local/code/inner.js @@ -1,71 +1,238 @@ -Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, target) { - if (msg.Result() == "" && msg.Length() == 0) { can.onmotion.hidden(can, can._output); return } - can.onmotion.toggle(can, can._output, true) +Volcanos(chat.ONIMPORT, {help: "导入数据", + _tabs: function(can) { + can.ui._tabs = can.page.insertBefore(can, ["tabs"], can.ui._content) + can.ui._path = can.page.insertBefore(can, ["path"], can.ui._content) + }, - if (msg.Option(nfs.FILE)) { - msg.Option(nfs.PATH) && can.Option(nfs.PATH, msg.Option(nfs.PATH)) - can.Option(nfs.FILE, msg.Option(nfs.FILE)) - msg.Option(nfs.LINE) && can.Option(nfs.LINE, msg.Option(nfs.LINE)) + project: function(can, path) { + can.onimport.zone(can, can.core.Item(can.onfigure, function(name, cb) { + return can.base.isFunc(cb)? {name: name, _init: function(target, zone) { return cb(can, target, zone, path) }}: undefined + }), can.ui.project), can.user.isMobile && !can.user.isLandscape() && can.onmotion.hidden(can, can.ui.project) + }, + tabview: function(can, path, file, line, cb) { var key = can.onexport.keys(can, path, file) + function isCommand() { return path == ctx.COMMAND || line == ctx.INDEX } + function isDream() { return line == web.DREAM } + + function show() { if (can.isCmdMode()) { can.onimport._title(can, path+file) } + can._msg && can._msg.Option(nfs.LINE, can.Option(nfs.LINE)), can._msg = can.tabview[key] + can.Option(can.onimport.history(can, {path: path, file: file, line: line||can._msg.Option(nfs.LINE)||1})) + can.onsyntax._init(can, can._msg, function(content) { var msg = can._msg + can.ui._path && (can.ui._path.innerHTML = isDream()? can.page.Format(html.A, can.misc.MergePodCmd(can, {pod: can.Option(nfs.FILE)})): + isCommand()? can.Option(nfs.FILE): can.base.Path(can.Option(nfs.PATH), can.Option(nfs.FILE))) + can.ui.current && can.onmotion.toggle(can, can.ui.current, !isCommand() && !isDream()) + + can.page.Select(can, can.ui._content.parentNode, can.page.Keys(html.DIV_CONTENT, html.IFRAME), function(item) { + if (can.onmotion.toggle(can, item, item == msg._content)) { can.ui.content = msg._content } + }) + can.page.Select(can, can.ui._profile_output.parentNode, can.page.Keys(html.DIV_OUTPUT, html.IFRAME), function(item) { + if (can.onmotion.toggle(can, item, item == msg._profile_output)) { can.ui.profile_output = msg._profile_output } + }) + can.onexport.hash(can), can.onimport.layout(can), can.base.isFunc(cb) && cb(), cb = null + isCommand() && can.onmotion.delay(can, function() { + can.onmotion.focus(can, can.page.Select(can, can.ui._content, html.OPTION_ARGS)[0]) + }), can.onengine.signal(can, "tabview.view.show", msg) + }) + } + function load(msg) { can.tabview[key] = msg + can.onimport.tabs(can, [{name: file.split(isCommand()? ice.PT: ice.PS).pop(), text: file}], function(event) { + can._tab = msg._tab = event.target, show() + }, function(item) { delete(can.tabview[key]) + can.onengine.signal(can, "tabview.view.delete", msg) + delete(can._cache_data[can.base.Path(path, file)]) + delete(can.ui._content._cache[can.base.Path(path, file)]) + delete(can.ui._profile_output._cache[can.base.Path(path, file)]) + delete(can.ui.display_output._cache[can.base.Path(path, file)]) + msg._content != can.ui._content && can.page.Remove(can, msg._content) + }, can.ui._tabs) } - var paths = can.core.Split(can.Option(nfs.PATH), ice.FS); can.Option(nfs.PATH, paths[0]) - var files = can.core.Split(can.Option(nfs.FILE), ice.FS); can.Option(nfs.FILE, files[0]) - can.core.List(paths.concat(msg.modules||[], can.sup.paths||[]), function(p) { if (paths.indexOf(p) == -1) { paths.push(p) } }) - can.sup.paths = paths + if (can.tabview[key]) { return can.isCmdMode() && !can._msg._tab? load(can.tabview[key]): show() } + isCommand()||isDream()? load(can.request({}, {index: file, line: line})): can.run({}, [path, file], load, true) + }, + history: function(can, push) { + can.base.Eq(push, can.history[can.history.length-1]) || can.history.push(push) + return can.Status("跳转数", can.history.length), push + }, +}, [""]) +Volcanos(chat.ONFIGURE, {help: "索引导航", + source: function(can, target, zone, path) { var total = 0 + function show(path, target) { can.run(can.request({}, {dir_root: path, dir_deep: true}), [ice.PWD], function(msg) { var list = msg.Table() + can.core.List(list, function(item) { if (can.Option(nfs.FILE).indexOf(item.path) == 0) { item.expand = true } }) + can.onimport.tree(can, list, nfs.PATH, ice.PS, function(event, item) { can.onimport.tabview(can, path, item.path) }, target) + can.Status("文件数", total += msg.Length()), zone._total(total) + }, true) } - can.tabview = can.tabview||{}, can.history = can.history||[], can.toolkit = {}, can.extentions = {} - can.profile_size = {}, can.display_size = {} + if (path.length == 1) { return show(path[0], target) } + can.onmotion.delay(can, function() { target.previousSibling.innerHTML = "" }) + can.onimport.zone(can, can.core.List(path, function(path) { return {name: path, _init: function(target) { show(path, target) }} }), target) + }, + plugin: function(can, target, zone) { var total = 0 + can.onimport.tree(can, can.core.Item(can.onengine.plugin.meta, function(key) { return total++, {index: can.base.trimPrefix(key, "can.")} }), ctx.INDEX, ice.PT, function(event, item) { + can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys("can", item.index), ctx.INDEX) + }, target), zone._total(total) + }, +}) +Volcanos(chat.ONSYNTAX, {help: "语法高亮", _init: function(can, msg, cb) { + if (can.onmotion.cache(can, function(cache_data) { + can.file && (cache_data[can.file] = {current: can.current, max: can.max, + profile_display: can.ui.profile.style.display, display_display: can.ui.display.style.display, + }) + can.file = can.base.Path(can.Option(nfs.PATH), can.Option(nfs.FILE)) + var p = cache_data[can.file]; p && (can.current = p.current, can.max = p.max) + can.page.style(can, can.ui.profile, {display: p? p.profile_display: html.NONE}) + can.page.style(can, can.ui.display, {display: p? p.display_display: html.NONE}) + can.parse = can.base.Ext(can.file), can.Status("模式", "plugin") + can.onengine.signal(can, "tabview.data.load", msg) + return can.file - can.page.ClassList.add(can, can._fields, code.INNER) - can.onmotion.clear(can), can.onlayout.profile(can) - can.onimport._project(can, can.ui.project) - can.onimport._profile(can, can.ui.profile) - can.onimport._display(can, can.ui.display) - - can.onimport._input(can) - can.require(["inner/syntax.js"], function() { can.base.isFunc(cb) && cb(msg) }) - can.ui._content = can.ui.content - can.ui._profile_output = can.ui.profile_output - can.onengine.plugin(can, can.onplugin) - - switch (can.Mode()) { - case "simple": can.onimport._simple(can); break - case "float": break - case "cmd": can.onimport._tabs(can), can.onmotion.hidden(can, can._status) // no break - case "full": // no break - default: can.onimport.project(can, paths) - can.onimport._toolkit(can, can.ui.toolkit), can.onimport._session(can, msg, function() { - files.length > 1 && can.onmotion.delay(can, function() { can.core.Next(files.slice(1), function(file, next) { - can.onimport.tabview(can, can.Option(nfs.PATH), file, can.Option(nfs.LINE), next) - }, function() { can.onimport.tabview(can, paths[0], files[0], "") }) }) - }), can.onimport._keydown(can) + }, can.ui._content, can.ui._profile_output, can.ui.display_output)) { + can.onengine.signal(can, "tabview.view.load", msg) + return can.onaction.selectLine(null, can, can.Option(nfs.LINE)), can.base.isFunc(cb) && cb(msg._content) } - var hash = location.hash; can.isCmdMode() || (can.tabview[can.onexport.keys(can)] = msg) - can.onimport.tabview(can, can.Option(nfs.PATH), can.Option(nfs.FILE), can.Option(nfs.LINE), function() { - if (can.isCmdMode() && hash) { var args = can.core.Split(decodeURIComponent(hash).slice(1)) - can.onmotion.delay(can, function() { can.onimport.tabview(can, can.Option(nfs.PATH), args[0], args[1]) }) + if (msg.Option(ctx.INDEX)) { return can.onsyntax._index(can, msg, cb) } + + function init(p) { + can.max = 0, can.core.List(can.ls = msg.Result().split(ice.NL), function(item) { can.onaction.appendLine(can, item) }) + can.onaction.selectLine(null, can, can.Option(nfs.LINE)), can.onengine.signal(can, "tabview.view.init", msg) + can.base.isFunc(cb) && cb(msg._content = can.ui._content) + } + can.require(["inner/syntax.js"], function() { can.Conf("plug") && (can.onsyntax[can.parse] = can.Conf("plug")) + var p = can.onsyntax[can.parse]; !p? can.runAction({}, mdb.PLUGIN, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)], function(msg) { + init(p = can.onsyntax[can.parse] = can.base.Obj(msg.Result())) + }): init(p) + }) + }, + _index: function(can, msg, cb) { + if (can.Option(nfs.LINE) == web.DREAM) { + return can.base.isFunc(cb) && cb(msg._content = msg._content||can.page.insertBefore(can, [{type: html.IFRAME, src: can.misc.MergePodCmd(can, {pod: can.Option(nfs.FILE)}), width: can.ui.content.offsetWidth, height: can.ui.content.offsetHeight}], can.ui._content)) + } + + return can.onimport.plug(can, {index: msg.Option(ctx.INDEX), args: can.Option(nfs.PATH) == ctx.COMMAND && can.Option(nfs.LINE) != ctx.INDEX? [can.Option(nfs.LINE)]: []}, can.ui._content, function(sub) { + can.page.style(can, sub._output, html.MAX_HEIGHT, sub.ConfHeight(can.ui._content.offsetHeight-2*html.ACTION_HEIGHT)) + can.page.style(can, sub._output, html.MAX_WIDTH, sub.ConfWidth(can.ui._content.offsetWidth)) + can.onmotion.delay(can, function() { sub.Focus() }), can.base.isFunc(cb) && cb(msg._content = can.ui._content) + + sub.onaction.close = function() { can.onaction.back({}, can), msg._tab._close() } + sub.onimport._open = function(sub, msg, _arg) { var url = can.base.ParseURL(_arg), ls = url.origin.split("/chat/pod/") + if (_arg.indexOf(location.origin) == 0 && ls.length > 1) { + return can.onimport.tabview(can, can.Option(nfs.PATH), ls[1].split("/")[0], web.DREAM), sub.Update() // 显示空间 + } + return can.user.open(_arg), sub.Update() } }) }, - _input: function(can) {}, - _project: function(can, target) { - target._toggle = function(event, show) { can.onimport.layout(can) } + _parse: function(can, line) { line = can.page.replace(can, line||"") + function wrap(type, str) { return can.page.Format(html.SPAN, str, type) } + var p = can.onsyntax[can.parse]; if (!p) { return line } p = can.onsyntax[p.link]||p, p.split = p.split||{} + p.keyword && (line = can.core.List(can.core.Split(line, p.split.space||"\t ", p.split.operator||"{[(,:;!|<*>)]}", {detail: true}), function(item, index, array) { + item = can.base.isObject(item)? item: {text: item}; var text = item.text, type = p.keyword[text] + switch (item.type||type) { + case html.SPACE: return text + case lang.STRING: return wrap(lang.STRING, item.left+text+item.right) + case code.COMMENT: + case code.KEYWORD: + case code.CONSTANT: + case code.DATATYPE: + case code.FUNCTION: return wrap(type, text) + default: + var t = can.core.Item(p.regexp, function(reg, type) { + var m = text.match(new RegExp(reg)); if (m && m.length > 0 && m[0] == text) { return type} + }); if (t && t.length > 0) { return wrap(t[0], text) } + return text + } + }).join("")) + p.prefix && can.core.Item(p.prefix, function(pre, type) { if (can.base.beginWith(line, pre)) { line = wrap(type, line) } }) + p.suffix && can.core.Item(p.suffix, function(end, type) { if (can.base.endWith(line, end)) { line = wrap(type, line) } }) + return line }, - _panel: function(can, target, action) { - var ui = can.page.Append(can, target, [html.ACTION, html.OUTPUT]) - var action = can.onappend._action(can, [], ui.action, kit.Dict( - cli.CLOSE, function(event) { can.onmotion.hidden(can, target), can.onimport.layout(can) }, - cli.CLEAR, function(event) { can.onmotion.clear(can, ui.output) }, - cli.SHOW, function(event) { can.onaction[cli.SHOW](event, can) }, - action, - mdb.PLUGIN, function(event) { can.user.input(event, can, [ctx.INDEX, ctx.ARGS], function(data) { can.onimport.plug(can, data, ui.output) }) }, - )); target._toggle = function(event, show) { action[show? cli.SHOW: cli.CLOSE](event) } - return ui +}) +Volcanos(chat.ONACTION, {help: "控件交互", list: [], + appendLine: function(can, value) { + var ui = can.page.Append(can, can.ui._content, [{type: html.TR, list: [ + {view: ["line unselectable", html.TD, ++can.max], onclick: function(event) { + can.onaction.selectLine(event, can, ui.tr) + }, ondblclick: function(event) { + can.onaction.favorLine(can, ui.text.innerText) + }}, + + {view: [html.TEXT, html.TD], inner: can.onsyntax._parse(can, value), onclick: function(event) { + can.onaction.selectLine(event, can, ui.tr) + + }, ondblclick: function(event) { + var s = document.getSelection().toString(), str = ui.text.innerText + var begin = str.indexOf(s), end = begin+s.length + for (var i = begin; i >= 0; i--) { if (str[i].match(/[a-zA-Z0-9_.]/)) { s = str.slice(i, end) } else { break } } + can.onaction.searchLine(event, can, s) + }} + ]}]); return ui.tr }, + selectLine: function(event, can, line) { if (!line) { return parseInt(can.core.Value(can.page.Select(can, can.ui._content, [[[html.TR, html.SELECT], [html.TD, "line"]]])[0], "innerText")) } + can.page.Select(can, can.ui._content, html.TR, function(item, index, array) { + if (!can.page.ClassList.set(can, item, html.SELECT, item == line || index+1 == line)) { return } + var ls = can.file.split(ice.PS); if (ls.length > 4) { ls = [ls.slice(0, 2).join(ice.PS)+"/.../"+ls.slice(-2).join(ice.PS)] } + line = item, can.Status(kit.Dict("文件名", ls.join(ice.PS), "解析器", can.parse, "当前行", can.onexport.position(can, can.Option(nfs.LINE, index+1)))) + }) + ; if (!can.base.isObject(line)) { return 1 } + + can.page.Select(can, line, "td.text", function(item) { + can.current = { + window: function() { return parseInt(can.ui._content.offsetHeight/can.current.line.offsetHeight) }, + scroll: function(count) { if (count) { can.ui._content.scrollTop += count*can.current.line.offsetHeight } + return parseInt((can.current.line.offsetTop-can.ui._content.scrollTop)/can.current.line.offsetHeight) + }, + line: line, text: function(text) { return text != undefined && can.onaction.modifyLine(can, line, text), item.innerText }, + prev: function() { return line.previousSibling }, next: function() { return line.nextSibling }, + } + + var scroll = can.current.scroll(); if (scroll < 3) { can.current.scroll(scroll-3) } else { + var window = can.current.window(); if (scroll > window-4) { can.current.scroll(scroll-window+4) } + } + + can.onimport.history(can, {path: can.Option(nfs.PATH), file: can.Option(nfs.FILE), line: can.Option(nfs.LINE)}) + can.onexport.hash(can), can.onengine.signal(can, "tabview.line.select", can.request(event)) + }); return parseInt(can.page.Select(can, line, "td.line")[0].innerText) + }, + searchLine: function(event, can, value) { + can.runAction(can.request(event, {name: value, text: can.current.text()}, can.Option()), NAVIGATE, [], function(msg) { + msg.Append(nfs.FILE)? can.onimport.tabview(can, msg.Append(nfs.PATH), msg.Append(nfs.FILE), msg.Append(nfs.LINE)): + can.user.toast(can, "not found") + }) + }, + favorLine: function(can, value) { + can.user.input(event, can, [{name: "zone", value: "hi"}, {name: "name", value: "hello"}], function(data) { + can.runAction(event, code.FAVOR, [ctx.ACTION, mdb.INSERT, mdb.ZONE, data.zone||"", + mdb.TYPE, can.parse, mdb.NAME, data.name||"", mdb.TEXT, (value||"").trimRight(), + nfs.PATH, can.Option(nfs.PATH), nfs.FILE, can.Option(nfs.FILE), nfs.LINE, can.Option(nfs.LINE), + ], function() { can.user.toast(can, ice.SUCCESS) }) + }) + }, + + back: function(event, can) { can.history.pop(); var last = can.history.pop() + last && can.onimport.tabview(can, last.path, last.file, last.line) + can.Status("跳转数", can.history.length) + }, +}) +Volcanos(chat.ONENGINE, {help: "搜索引擎", + listen: shy("监听事件", function(can, name, cb) { arguments.callee.meta[name] = (arguments.callee.meta[name]||[]).concat(cb) }), +}) +Volcanos(chat.ONEXPORT, {help: "导出数据", list: ["文件数", "解析器", "文件名", "当前行", "跳转数", "标签数"], + keys: function(can, path, file, line) { + return [path||can.Option(nfs.PATH), file||can.Option(nfs.FILE)].join(ice.DF) + }, + hash: function(can) { if (!can.isCmdMode()) { return } + var list = []; if (can.Option(nfs.PATH) != can.misc.Search(can, nfs.PATH)) { list.push(can.Option(nfs.PATH)) } + if (list.length > 0 || can.Option(nfs.FILE) != can.misc.Search(can, nfs.FILE)) { list.push(can.Option(nfs.FILE)) } + if (list.length > 0 || can.Option(nfs.LINE) != can.misc.Search(can, nfs.LINE)) { list.push(can.Option(nfs.LINE)||1) } + location.hash = list.join(ice.FS) + }, + content: function(can) { return can.page.Select(can, can.ui.content, "td.text", function(item) { return item.innerText }).join(ice.NL) }, + position: function(can, index, total) { total = total||can.max; return (parseInt(index))+ice.PS+parseInt(total)+" = "+parseInt((index)*100/total)+"%" }, +}) + +Volcanos(chat.ONIMPORT, {help: "导入数据", _profile: function(can, target) { var ui = can.onimport._panel(can, target, kit.Dict( - nfs.LOAD, function(event) { can.onaction[nfs.LOAD](event, can) }, mdb.LINK, function(event) { if ([nfs.ZML, nfs.IML].indexOf(can.base.Ext(can.Option(nfs.FILE))) > -1) { can.user.open(can.misc.MergeURL(can, { @@ -88,7 +255,6 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar _display: function(can, target) { var ui = can.onimport._panel(can, target, kit.Dict( cli.SHOW, function(event) { can.onaction[cli.EXEC](event, can) }, - mdb.PLUGIN, function(event) { can.user.input(event, can, [ctx.INDEX, ctx.ARGS], function(data) { can.onimport.plug(can, data, ui.output) }) }, html.HEIGHT, function(event) { can.user.input(event, can, [{name: html.HEIGHT, value: can.display_size[can.onexport.keys(can)]*100/can.ConfHeight()||50}], function(list) { can.display_size[can.onexport.keys(can)] = can.ConfHeight()*parseInt(list[0])/100 @@ -97,116 +263,21 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar } )); can.ui.display_output = ui.output, can.ui.display_status = ui.status }, - _toolkit: function(can, target) { - can.ui.toolkit = can.onappend.field(can, "toolkit", {}, can._output) - }, - _session: function(can, msg) { - can.onimport.sess(can, "", function() { can.onimport.sess(can, { - exts: can.core.Split(msg.OptionOrSearch("exts")).reverse(), - plug: can.core.Split(msg.OptionOrSearch("plug")).reverse(), - tabs: can.core.Split(msg.OptionOrSearch("tabs")), - }) }) - }, - _keydown: function(can) { can.onkeymap._build(can) - can.isCmdMode() && can.onengine.listen(can, chat.ONKEYDOWN, function(event) { - can._key_list = can.onkeymap._parse(event, can, mdb.PLUGIN, can._key_list, can.ui.content) - }) - }, - _simple: function(can) { - can.tabview[can.onexport.keys(can)] = can._msg - can.ConfHeight(""), can.onmotion.hidden(can, can.ui.project) - can.page.ClassList.add(can, can._fields, html.OUTPUT) - can.page.ClassList.add(can, can._fields, "simple") - }, - _tabs: function(can) { - can.ui._tabs = can.page.insertBefore(can, [{view: "tabs"}], can.ui.content) - can.ui._path = can.page.insertBefore(can, [{view: "path"}], can.ui.content) - }, - - layout: function(can) { if (can.isSimpleMode()) { return } - if (can.isFloatMode()) { can.onmotion.hidden(can, can._action), can.onmotion.hidden(can, can.ui.project) } - - var width = can.ConfWidth()+(can.user.isMobile && can.isCmdMode() && can.user.isLandscape()? 16: 0)-(can.user.isWindows && !can.isCmdMode()? 20: 0) - can.page.styleWidth(can, can.ui.profile_output, can.profile_size[can.onexport.keys(can)]||(width-can.ui.project.offsetWidth)/3) - can.page.styleWidth(can, can.ui.content, width-can.ui.project.offsetWidth-can.ui.profile.offsetWidth) - can.page.styleWidth(can, can.ui.display, width-can.ui.project.offsetWidth) - - var height = can.ConfHeight() - if (can.user.isMobile && can.isFloatMode()) { height = can._root._height-2*html.ACTION_HEIGHT } - - if (can.isCmdMode()) { - if (can.ui.display.display != html.NONE) { - if (can.ui.display.offsetHeight > can.base.Min(can.ConfHeight() / 2, 200)) { - can.page.style(can, can.ui.display_output, html.MAX_HEIGHT, can.base.Min(can.ConfHeight() / 2, 200)) - } - } - if (can._status.style.display == html.NONE) { height += html.ACTION_HEIGHT } - var rest = can.ui.display.offsetHeight+can.ui._tabs.offsetHeight+can.ui._path.offsetHeight+4 - can.page.styleHeight(can, can.ui.content, height+html.ACTION_HEIGHT-rest) - can.page.styleHeight(can, can.ui.profile_output, height-can.ui.display.offsetHeight) - can.page.styleHeight(can, can.ui.project, height+html.ACTION_HEIGHT) - } else { var rest = can.ui.display.offsetHeight; if (height < 320) { height = 320 } - if (!can.isFullMode()) { - can._min_height = can._min_height||height, height >= can._min_height && (can._min_height = height) - can.page.style(can, can.ui.content, html.MIN_HEIGHT, can._min_height) - } - can.page.style(can, can.ui.content, can.user.isMobile? html.HEIGHT: html.MAX_HEIGHT, height-rest) - can.page.styleHeight(can, can.ui.profile_output, can.ui.content.offsetHeight-html.ACTION_HEIGHT) - can.page.styleHeight(can, can.ui.project, can.ui.content.offsetHeight+rest) - } - can.page.Select(can, can.ui.profile, html.IFRAME, function(iframe) { - can.page.Modify(can, iframe, {height: can.ui.profile.offsetHeight-html.ACTION_HEIGHT-4, width: can.ui.profile.offsetWidth}) - }) - }, - project: function(can, path) { - if (can.ConfHeight() < 320) { can.ConfHeight(320) } - can.user.isMobile && !can.user.isLandscape() && can.onmotion.hidden(can, can.ui.project) - can.onimport.zone(can, can.core.Item(can.onfigure, function(name, cb) { - return can.base.isFunc(cb)? {name: name, _init: function(target, zone) { cb(can, target, zone, path) }}: undefined - }), can.ui.project) - }, - history: function(can, push) { - can.base.Eq(push, can.history[can.history.length-1]) || can.history.push(push) - return can.Status("跳转数", can.history.length), push - }, - tabview: function(can, path, file, line, cb, skip, skip2) { var key = can.onexport.keys(can, file, path) - if (can.isCmdMode()) { (location.hash = location.pathname.indexOf(file) > -1? "": file+ice.FS+(line||1)) } - - if (!skip && can.tabview[key]) { if (can.isCmdMode()) { can.onimport._title(can, path+file) } - can._msg && can._msg.Option(nfs.LINE, can.Option(nfs.LINE)), can._msg = can.tabview[key] - can.Option({path: path, file: file, line: line||can._msg.Option(nfs.LINE)||1}) - return can._msg.Option(can.Option()), can.onsyntax._init(can, can._msg, cb, skip2) - } - - function show(msg) { var skip2 = skip; can.tabview[key] = msg - can.onimport.tabs(can, [{name: file.split(line == ctx.INDEX? ice.PT: ice.PS).pop(), text: file}], function(event, meta) { - can._tab = msg._tab = event.target, can.onimport.tabview(can, path, file, msg.Option(nfs.LINE)||line, cb, false, skip2), cb = null, skip2 = false - }, function(item) { // 关闭文件 - delete(can.tabview[key]) - delete(can._cache_data[can.base.Path(path, file)]) - delete(can.ui._content._cache[can.base.Path(path, file)]) - delete(can.ui._profile_output._cache[can.base.Path(path, file)]) - delete(can.ui.display_output._cache[can.base.Path(path, file)]) - msg._content != can.ui._content && can.page.Remove(can, msg._content) - }, can.ui._tabs, function(item) {}) - } - - can.Option(can.onimport.history(can, {path: path, file: file, line: line})) - path == ctx.COMMAND || line == ctx.INDEX? /* 命令 */ show(can.request({}, {index: file, line: line})): - line == web.DREAM? /* 空间 */ show(can.request({}, {index: file, line: line})): - /* 文件 */ can.run({}, [path, file], show, true) - }, profile: function(can, msg) { - var width = can.profile_size[can.onexport.keys(can)]||(can.ConfWidth()-can.ui.project.offsetWidth)/2 if (msg) { var sup = can.tabview[can.onexport.keys(can)] can.onmotion.toggle(can, can.ui.profile_output, false) if (msg.Result().indexOf(" -1) { + var width = can.profile_size[can.onexport.keys(can)]||(can.ConfWidth()-can.ui.project.offsetWidth)/4*3 if (sup._profile_output != can.ui._profile_output) { can.page.Remove(can, sup._profile_output) } - can.ui.profile_output = sup._profile_output = can.page.Append(can, can.ui.profile_output.parentNode, [{view: "output", inner: msg.Result()}]).output + can.ui.profile_output = sup._profile_output = can.page.Append(can, can.ui._profile_output.parentNode, [{view: "output", inner: msg.Result()}]).output + can.profile_size[can.onexport.keys(can)] = width } else { + var width = can.profile_size[can.onexport.keys(can)]||(can.ConfWidth()-can.ui.project.offsetWidth)/3 can.ui.profile_output = sup._profile_output = can.ui._profile_output can.onimport.process(can, msg, can.ui._profile_output, width, can.ui.profile.offsetHeight) + can.onappend._status(can, msg.Option(ice.MSG_STATUS), can.page.Append(can, can.ui._profile_output, ["status"]).first) + can.page.Select(can, can.ui._profile_output, "table.content", function(target) { can.page.style(can, target, "max-height", "1000px") }) } } can.onmotion.toggle(can, can.ui.profile_output, true) @@ -222,8 +293,8 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar }, process: function(can, msg, target, width, height) { can.onmotion.clear(can, target), can.user.toastSuccess(can) if (msg.Option(ice.MSG_PROCESS) == "_field") { - msg.Table(function(meta) { meta.display = msg.Option(ice.MSG_DISPLAY) - can.onimport.plug(can, meta, target, function(sub) { width && sub.ConfWidth(width), height && sub.ConfHeight(height), sub.Focus() }) + msg.Table(function(item) { item.display = msg.Option(ice.MSG_DISPLAY) + can.onimport.plug(can, item, target, function(sub) { width && sub.ConfWidth(width), height && sub.ConfHeight(height), sub.Focus() }) }) } else if (msg.Option(ice.MSG_DISPLAY) != "") { can.onappend._output(can, msg, msg.Option(ice.MSG_DISPLAY), target, false, function(msg) { can.onmotion.delay(can, function() { can.onimport.layout(can) }) }) @@ -231,180 +302,59 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar can.onappend.table(can, msg, null, target), can.onappend.board(can, msg, target) } }, - toolkit: function(can, meta, cb) { meta.msg = true - can.onimport.plug(can, meta, can.ui.toolkit.output, function(sub) { - sub.ConfHeight(can.ConfHeight()-4*html.ACTION_HEIGHT), sub.ConfWidth(can.ConfWidth()) - sub.page.style(sub, sub._output, html.MAX_HEIGHT, sub.ConfHeight()) - sub.page.style(sub, sub._output, html.MAX_WIDTH, sub.ConfWidth()) - - can._status.appendChild(sub._legend), sub._legend.onclick = function(event) { - if (can.page.Select(can, can._status, ice.PT+html.SELECT)[0] == event.target) { - can.page.ClassList.del(can, event.target, html.SELECT) - can.page.ClassList.del(can, sub._target, html.SELECT) - return - } - can.onmotion.select(can, can.ui.toolkit.output, html.FIELDSET, sub._target), sub.Focus() - can.onmotion.select(can, can._status, html.LEGEND, event.target) - if (meta.msg == true) { meta.msg = false, sub.Update() } - }, sub.select = function() { return sub._legend.click(), sub } - sub.onaction.close = function() { sub.select() } - sub._legend.onmouseenter = null - can.base.isFunc(cb) && cb(sub) - }) - }, - exts: function(can, url, cb) { - can.require([url], function() {}, function(can, name, sub) { sub._init(can, can.base.ParseURL(sub._path), function(sub) { - can.extentions[url.split("?")[0]] = sub, can.base.isFunc(cb) && cb(sub) - }) }) - }, - sess: function(can, sess, cb) { sess = sess||can.user.localStorage(can, "web.code.inner.sess") - can.core.Next(sess.plug, function(item, next) { can.onimport.toolkit(can, {index: item}, function(sub) { can.toolkit[item] = sub, next() }) }, function() { - can.core.Next(sess.exts, function(item, next) { can.onimport.exts(can, item, next) }, function() { - var path = can.Option(nfs.PATH), file = can.Option(nfs.FILE), line = can.Option(nfs.LINE) - can.base.getValid(sess.tabs)? can.core.Next(sess.tabs, function(item, next) { var ls = item.split(ice.DF); can.onimport.tabview(can, ls[0], ls[1], ls[2], next) }, - function() { can.onimport.tabview(can, path, file, line, cb) }): can.base.isFunc(cb) && cb() - }) - }) - }, }, [""]) -Volcanos(chat.ONFIGURE, {help: "导航索引", - source: function(can, target, zone, path) { var total = 0 - function show(path, target) { can.run(can.request({}, {dir_root: path, dir_deep: true}), [ice.PWD], function(msg) { var list = msg.Table() - can.core.List(list, function(item) { if (can.Option(nfs.FILE).indexOf(item.path) == 0) { item.expand = true } }) - can.onimport.tree(can, list, nfs.PATH, ice.PS, function(event, item) { - can.onimport.tabview(can, path, item.path) // 显示文件 - }, target), can.Status("文件数", total += msg.Length()) - }, true) } if (path.length == 1) { return show(path[0], target) } - - can.onimport.zone(can, can.core.List(path, function(path) { return {name: path, _init: function(target) { show(path, target) }} }), target) - can.onmotion.delay(can, function() { target.previousSibling.innerHTML = "" }) +Volcanos(chat.ONACTION, {help: "控件交互", _trans: {link: "链接", width: "宽度", height: "高度"}, + "搜索": function(event, can) { + can.user.input(event, can, [mdb.NAME, [ctx.ACTION, nfs.TAGS, nfs.GREP, cli.MAKE]], function(data) { + can.ui.search.Update({}, [ctx.ACTION, data.action, data.name]) + }) }, - plugin: function(can, target, zone) { - can.onimport.tree(can, can.core.Item(can.onengine.plugin.meta, function(key) { return {index: can.base.trimPrefix(key, "can.")} }), ctx.INDEX, ice.PT, function(event, item) { - can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys("can", item.index), ctx.INDEX) // 显示插件 - }, target) + "打开": function(event, can) { + can.user.input(event, can, [nfs.FILE], function(list) { can.onimport.tabview(can, can.Option(nfs.PATH), list[0]) }) + }, + show: function(event, can) { can.request(event, {_toast: "渲染中..."}) + if (can.base.endWith(can.Option(nfs.FILE), ".js")) { + var file = can.base.Path("/require/", can.Option(nfs.PATH), can.Option(nfs.FILE)) + delete(Volcanos.meta.cache[file]), eval("\n_can_name = \""+file+"\"\n"+can.onexport.content(can)+"\n_can_name = \"\"\nconsole.log(\"once\")") + } + can.runAction(event, mdb.RENDER, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)], function(msg) { + can.onimport.profile(can, msg) + }) + }, + exec: function(event, can) { can.request(event, {_toast: "执行中...", "some": "run"}) + can.runAction(event, mdb.ENGINE, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)], function(msg) { + can.onimport.display(can, msg) + }) + }, + clear: function(event, can) { + if (can.page.Select(can, can._root._target, ".input.float", function(item) { return can.page.Remove(can, item) }).length > 0) { return } + if (can.page.Select(can, can._status, "legend.select", function(item) { return item.click(), item }).length > 0) { return } + + if (can.ui.display.style.display == "") { return can.onmotion.hidden(can, can.ui.display), can.onimport.layout(can) } + if (can.ui.profile.style.display == "") { return can.onmotion.hidden(can, can.ui.profile), can.onimport.layout(can) } + can.onmotion.toggle(can, can.ui.project), can.onimport.layout(can) + }, + listTags: function(event, can, button) { var list = [] + can.core.Item(can.request(event), function(key, value) { if (key.indexOf("_") == 0) { return } + list.push({zone: "msg", type: typeof value, name: key, text: can.base.isObject(value)? "": (value+"").split(ice.NL)[0], + path: "usr/volcanos/", file: "lib/misc.js", line: 1, + }) + }) + can.core.List([can.base, can.core, can.misc, can.page, can.user, can.onengine, can.ondaemon, can.onappend, can.onlayout, can.onmotion, can.onkeymap], function(lib) { + can.core.Item(lib, function(key, value) { if (key.indexOf("_") == 0 || !lib.hasOwnProperty(key)) { return } + list.push({zone: lib._name, type: typeof value, name: key, text: can.base.isObject(value)? "": (value+"").split(ice.NL)[0], + path: "usr/volcanos/", file: lib._path, line: 1, + }) + }) + }) + can.runAction(can.request(event, {text: can.base.Format(list)}), button) }, }) -Volcanos(chat.ONPLUGIN, {help: "注册插件", - "can.code.inner.keymap": shy("按键", {}, ["mode", "key", ice.LIST, ice.BACK], function(can, msg, cmds) { - can.core.Item(can.onkeymap._mode, function(mode, value) { - (!cmds[0] || cmds[0] == mode) && can.core.Item(value, function(key, func) { - if (cmds[0] == mode && cmds[1] == key) { - msg.Push("key", "mode") - msg.Push("value", mode) - msg.Push("key", "key") - msg.Push("value", key) - msg.Push("key", "help") - msg.Push("value", func.help) - msg.Push("key", "func") - msg.Push("value", func.toString()) - } else if (!cmds[0] || !cmds[1]) { - msg.Push(kit.Dict("mode", mode, "key", key, "help", func.help||func.toString())) - } - }) - }), msg.StatusTimeCount() - }), -}) -Volcanos(chat.ONSYNTAX, {help: "语法高亮", list: ["keyword", "prefix", "line"], _init: function(can, msg, cb, skip) { - if (can.ui._path) { - can.ui._path.innerHTML = msg.Option(ctx.INDEX)? msg.Option(nfs.LINE) == web.DREAM? - /* 空间 */ can.page.Format(html.A, can.misc.MergeURL(can, {pod: msg.Option(nfs.FILE), topic: can.misc.Search(can, "topic")}, true)): - /* 命令 */ msg.Option(nfs.FILE): /* 文件 */ can.base.Path(msg.Option(nfs.PATH), msg.Option(nfs.FILE)) - } - can.ui.current && can.onmotion.toggle(can, can.ui.current, !msg.Option(ctx.INDEX)) - - if (can.onmotion.cache(can, function(cache_data) { - can.file && (cache_data[can.file] = {current: can.current, max: can.max, - profile_display: can.ui.profile.style.display, display_display: can.ui.display.style.display, - }) - can.file = can.base.Path(msg.Option(nfs.PATH), msg.Option(nfs.FILE)) - can.parse = can.base.Ext(can.file), can.Status("模式", "plugin") - - var p = cache_data[can.file]; p && (can.current = p.current, can.max = p.max) - can.page.style(can, can.ui.profile, {display: p? p.profile_display: html.NONE}) - can.page.style(can, can.ui.display, {display: p? p.display_display: html.NONE}) - can.onmotion.select(can, can.ui._tabs, html.DIV_TABS, msg._tab) - can.onmotion.select(can, can._action, html.DIV_TABS, msg._tab) - can.onmotion.delay(can, function() { can.onimport.layout(can) - msg.Option(ctx.INDEX) && can.onmotion.focus(can, can.page.Select(can, can.ui.content, html.OPTION_ARGS)[0]) - }) - - can.page.Select(can, can.ui._content.parentNode, can.page.Keys("div.content", html.IFRAME), function(item) { - if (can.onmotion.toggle(can, item, item == msg._content)) { can.ui.content = msg._content } - }) - can.page.Select(can, can.ui._profile_output.parentNode, can.page.Keys("div.output"), function(item) { - if (can.onmotion.toggle(can, item, item == msg._profile_output)) { msg._profile_output } - }) - return can.file - }, can.ui._content, can.ui._profile_output, can.ui.display_output) && !skip) { - return can.onaction.selectLine(null, can, msg.Option(nfs.LINE)), can.base.isFunc(cb) && cb() - } - - can.onmotion.clear(can, can.ui.content), can.onimport.layout(can) - if (msg.Option(ctx.INDEX)) { - if (msg.Option(nfs.LINE) == web.DREAM) { if (msg._content) { return } - can.ui.content = msg._content = can.page.insertBefore(can, [{type: html.IFRAME, src: can.misc.MergeURL(can, {pod: msg.Option(nfs.FILE), topic: can.misc.Search(can, "topic")}, true), width: can.ui.content.offsetWidth, height: can.ui.content.offsetHeight}], can.ui.content) - return can.onimport.layout(can) - } - - can.onmotion.toggle(can, can.ui.content = msg._content = can.ui._content, true) - return can.onimport.plug(can, {index: msg.Option(ctx.INDEX), args: msg.Option(nfs.PATH) == ctx.COMMAND? [msg.Option(nfs.LINE)]: []}, can.ui.content, function(sub) { - can.page.style(can, sub._output, html.MAX_HEIGHT, sub.ConfHeight(can.ui.content.offsetHeight-2*html.ACTION_HEIGHT)) - can.page.style(can, sub._output, html.MAX_WIDTH, sub.ConfWidth(can.ui.content.offsetWidth)) - can.onmotion.delay(can, function() { sub.Focus() }), can.base.isFunc(cb) && cb() - - sub.onaction.close = function() { can.onaction.back({}, can), msg._tab._close() } - sub.onimport._open = function(sub, msg, _arg) { var url = can.base.ParseURL(_arg), ls = url.origin.split("/chat/pod/") - if (_arg.indexOf(location.origin) == 0 && ls.length > 1) { - return can.onimport.tabview(can, can.Option(nfs.PATH), ls[1].split("/")[0], web.DREAM), sub.Update() // 显示空间 - } - return can.user.open(_arg), sub.Update() - } - }) - } - can.onmotion.toggle(can, can.ui.content = msg._content = can.ui._content, true) - - function init(p) { - can.max = 0, can.core.List(can.ls = msg.Result().split(ice.NL), function(item) { can.onaction.appendLine(can, item) }) - can.onaction.selectLine(null, can, msg.Option(nfs.LINE)), can.base.isFunc(cb) && cb() - - msg.Option(nfs.FILE).indexOf("website/") == 0 && can.onaction[cli.SHOW]({}, can) - p && p.render && can.onaction[cli.SHOW]({}, can) - if (can.page.ClassList.has(can, can._fields, chat.PLUGIN)) { - p && p.engine && can.onaction[cli.EXEC]({}, can) - } - } - can.Conf("plug") && (can.onsyntax[can.parse] = can.Conf("plug")) - var p = can.onsyntax[can.parse]; !p? can.runAction({}, mdb.PLUGIN, [can.parse, msg.Option(nfs.FILE), msg.Option(nfs.PATH)], function(msg) { - init(p = can.onsyntax[can.parse] = can.base.Obj(msg.Result())) - }): init(p) - }, - _parse: function(can, line) { if (line.indexOf("'+str+'': str } - var p = can.onsyntax[can.parse]; if (!p) { return line } p = can.onsyntax[p.link]||p, p.split = p.split||{} - p.keyword && (line = can.core.List(can.core.Split(line, p.split.space||"\t ", p.split.operator||"{[(,:;!|<*>)]}", {detail: true}), function(item, index, array) { - item = can.base.isObject(item)? item: {text: item}; var text = item.text, type = item.keyword||p.keyword[text] - switch (item.type) { - case html.SPACE: return text - case lang.STRING: return wrap(lang.STRING, item.left+text+item.right) - case code.COMMENT: - case code.KEYWORD: - case code.CONSTANT: - case code.DATATYPE: - case code.FUNCTION: return wrap(type, text) - default: - var t = can.core.Item(p.regexp, function(reg, type) { - var m = text.match(new RegExp(reg)) - if (m && m.length > 0 && m[0] == text) { return type} - }); if (t && t.length > 0) { return wrap(t[0], text) } - return wrap(type, text) - } - }).join("")) - p.prefix && can.core.Item(p.prefix, function(pre, type) { if (can.base.beginWith(line, pre)) { line = wrap(type, line) } }) - p.suffix && can.core.Item(p.suffix, function(end, type) { if (can.base.endWith(line, end)) { line = wrap(type, line) } }) - return line +Volcanos(chat.ONIMPORT, {help: "导入数据", + _keydown: function(can) { can.onkeymap._build(can) + can.isCmdMode() && can._root.onengine.listen(can, chat.ONKEYDOWN, function(event) { + can._key_list = can.onkeymap._parse(event, can, mdb.PLUGIN, can._key_list, can.ui.content) + }) }, }) Volcanos(chat.ONKEYMAP, {help: "导入数据", @@ -438,149 +388,148 @@ Volcanos(chat.ONKEYMAP, {help: "导入数据", }, }, _engine: {}, }) -Volcanos(chat.ONACTION, {help: "控件交互", list: ["搜索", "打开"], - _trans: {source: "源码", module: "模块", dreams: "空间", load: "加载", link: "链接", width: "宽度", height: "高度", website: "网页"}, - "搜索": function(event, can) { - can.user.input(event, can, [mdb.NAME, [ctx.ACTION, nfs.TAGS, nfs.GREP, cli.MAKE]], function(data) { - can.ui.search.Update({}, [ctx.ACTION, data.action, data.name]) + +Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, target) { + can.page.ClassList.add(can, can._fields, code.INNER) + can.onmotion.clear(can), can.onlayout.profile(can) + can.onimport._profile(can, can.ui.profile) + can.onimport._display(can, can.ui.display) + if (msg.Result() == "") { return } + + if (msg.Option(nfs.FILE)) { + msg.Option(nfs.PATH) && can.Option(nfs.PATH, msg.Option(nfs.PATH)) + can.Option(nfs.FILE, msg.Option(nfs.FILE)) + msg.Option(nfs.LINE) && can.Option(nfs.LINE, msg.Option(nfs.LINE)) + } + + var paths = can.core.Split(can.Option(nfs.PATH), ice.FS); can.Option(nfs.PATH, paths[0]) + var files = can.core.Split(can.Option(nfs.FILE), ice.FS); can.Option(nfs.FILE, files[0]) + can.core.List(paths.concat(msg.modules||[], can.sup.paths||[]), function(p) { if (paths.indexOf(p) == -1) { paths.push(p) } }) + can.sup.paths = paths + + can.ui._content = can.ui.content, can.ui._profile_output = can.ui.profile_output + can.tabview = can.tabview||{}, can.history = can.history||[], can.toolkit = {}, can.extentions = {} + can.profile_size = {}, can.display_size = {} + can.base.isFunc(cb) && cb(msg) + + can.onengine.listen(can, "tabview.view.init", function() { var p = can.onsyntax[can.parse] + if (can.Mode() == "simple") { return } + p && p.render && can.onaction[cli.SHOW]({}, can); if (can.page.ClassList.has(can, can._fields, chat.PLUGIN)) { + p && p.engine && can.onaction[cli.EXEC]({}, can) + } + }) + + switch (can.Mode()) { + case "simple": can.onmotion.hidden(can, can.ui.project); break + case "float": break + case "cmd": can.onimport._tabs(can), can.onmotion.hidden(can, can._status) // no break + case "full": // no break + default: can.onimport.project(can, paths) + can.onimport._keydown(can), can.onimport._toolkit(can, can.ui.toolkit), can.onimport._session(can, msg, function() { + files.length > 1 && can.onmotion.delay(can, function() { can.core.Next(files.slice(1), function(file, next) { + can.onimport.tabview(can, can.Option(nfs.PATH), file, can.Option(nfs.LINE), next) + }, function() { can.onimport.tabview(can, paths[0], files[0], "") }) }) + }) + } + + var hash = location.hash; can.tabview[can.onexport.keys(can)] = msg + can.onimport.tabview(can, can.Option(nfs.PATH), can.Option(nfs.FILE), can.Option(nfs.LINE), function() { + if (can.isCmdMode() && hash) { var args = can.core.Split(decodeURIComponent(hash).slice(1)) + can.onmotion.delay(can, function() { can.onimport.tabview(can, args[args.length-3]||can.Option(nfs.PATH), args[args.length-2]||can.Option(nfs.FILE), args[args.length-1]) }) + } }) }, - "打开": function(event, can) { - can.user.input(event, can, [nfs.FILE], function(list) { can.onimport.tabview(can, can.Option(nfs.PATH), list[0]) }) + _toolkit: function(can, target) { + can.ui.toolkit = can.onappend.field(can, "toolkit", {}, can._output) }, + _session: function(can, msg) { + can.onimport.sess(can, "", function() { can.onimport.sess(can, { + exts: can.core.Split(msg.OptionOrSearch("exts")).reverse(), + plug: can.core.Split(msg.OptionOrSearch("plug")).reverse(), + tabs: can.core.Split(msg.OptionOrSearch("tabs")), + }) }) + }, + layout: function(can) { if (can.isSimpleMode()) { + return can.page.style(can, can.ui.content, html.WIDTH, can.ConfWidth()) + } + if (can.isFloatMode()) { can.onmotion.hidden(can, can._action), can.onmotion.hidden(can, can.ui.project) } + + var height = can.user.isMobile && can.isFloatMode()? window.innerHeight-2*html.ACTION_HEIGHT: can.ConfHeight() + var width = can.ConfWidth()+(can.user.isMobile && can.isCmdMode() && can.user.isLandscape()? 16: 0)-(can.user.isWindows && !can.isCmdMode()? 20: 0) + can.page.styleWidth(can, can.ui.profile_output, can.profile_size[can.onexport.keys(can)]||(width-can.ui.project.offsetWidth)/3) + can.page.styleWidth(can, can.ui.content, width-can.ui.project.offsetWidth-can.ui.profile.offsetWidth) + can.page.styleWidth(can, can.ui.display, width-can.ui.project.offsetWidth) + + var displayHeight = can.display_size[can.onexport.keys(can)] + can.page.style(can, can.ui.display, html.HEIGHT, displayHeight||200) + + if (can.isCmdMode()) { + if (can.ui.display.display != html.NONE) { + if (can.ui.display.offsetHeight > can.base.Min(can.ConfHeight() / 2, 200)) { + can.page.style(can, can.ui.display_output, html.MAX_HEIGHT, can.base.Min(can.ConfHeight() / 2, 200)) + } + } + if (can._status.style.display == html.NONE) { height += html.ACTION_HEIGHT } + var rest = can.ui.display.offsetHeight+can.ui._tabs.offsetHeight+can.ui._path.offsetHeight+4 + can.page.styleHeight(can, can.ui.content, height+html.ACTION_HEIGHT-rest) + can.page.styleHeight(can, can.ui.profile_output, height-can.ui.display.offsetHeight) + can.page.styleHeight(can, can.ui.project, height+html.ACTION_HEIGHT) + } else { var rest = can.ui.display.offsetHeight; if (height < 320) { height = 320 } + if (!can.isFullMode()) { + can._min_height = can._min_height||height, height >= can._min_height && (can._min_height = height) + can.page.style(can, can.ui.content, html.MIN_HEIGHT, can._min_height) + } + can.page.style(can, can.ui.content, can.user.isMobile? html.HEIGHT: html.MAX_HEIGHT, height-rest) + can.page.styleHeight(can, can.ui.profile_output, can.ui.content.offsetHeight-html.ACTION_HEIGHT) + can.page.styleHeight(can, can.ui.project, can.ui.content.offsetHeight+rest) + } + can.page.Select(can, can.ui.profile, html.IFRAME, function(iframe) { + can.page.Modify(can, iframe, {height: can.ui.profile.offsetHeight-html.ACTION_HEIGHT-4, width: can.ui.profile.offsetWidth}) + }) + }, + toolkit: function(can, meta, cb) { meta.msg = true + can.onimport.plug(can, meta, can.ui.toolkit.output, function(sub) { + sub.ConfHeight(can.ConfHeight()/2), sub.ConfWidth(can.ui.content.offsetWidth+can.ui.profile.offsetWidth) + sub.page.style(sub, sub._output, html.MAX_HEIGHT, sub.ConfHeight()) + sub.page.style(sub, sub._output, html.MAX_WIDTH, sub.ConfWidth()) + + can._status.appendChild(sub._legend), sub._legend.onclick = function(event) { + if (can.page.Select(can, can._status, ice.PT+html.SELECT)[0] == event.target) { + can.page.ClassList.del(can, event.target, html.SELECT) + can.page.ClassList.del(can, sub._target, html.SELECT) + return + } + + can.onmotion.select(can, can.ui.toolkit.output, html.FIELDSET, sub._target), sub.Focus() + can.onmotion.select(can, can._status, html.LEGEND, event.target) + if (meta.msg == true) { meta.msg = false, sub.Update() } + }, sub.select = function() { return sub._legend.click(), sub } + sub.onaction.close = function() { sub.select() } + sub._legend.onmouseenter = null + can.base.isFunc(cb) && cb(sub) + }) + }, + exts: function(can, url, cb) { + can.require([url], function() {}, function(can, name, sub) { sub._init(can, can.base.ParseURL(sub._path), function(sub) { + can.extentions[url.split("?")[0]] = sub, can.base.isFunc(cb) && cb(sub) + }) }) + }, + sess: function(can, sess, cb) { sess = sess||can.user.localStorage(can, "web.code.inner.sess") + can.core.Next(sess.plug, function(item, next) { can.onimport.toolkit(can, {index: item}, function(sub) { can.toolkit[item] = sub, next() }) }, function() { + can.core.Next(sess.exts, function(item, next) { can.onimport.exts(can, item, next) }, function() { + var path = can.Option(nfs.PATH), file = can.Option(nfs.FILE), line = can.Option(nfs.LINE) + can.base.getValid(sess.tabs)? can.core.Next(sess.tabs, function(item, next) { var ls = item.split(ice.DF); can.onimport.tabview(can, ls[0], ls[1], ls[2], next) }, + function() { can.onimport.tabview(can, path, file, line, cb) }): can.base.isFunc(cb) && cb() + }) + }) + }, +}, [""]) +Volcanos(chat.ONACTION, {help: "控件交互", list: [], sess: function(event, can) { can.onexport.sess(can), can.user.toastSuccess(can) }, - load: function(event, can) { - var file = can.base.Path("/require/", can.Option(nfs.PATH), can.Option(nfs.FILE)) - delete(Volcanos.meta.cache[file]), eval("\n_can_name = \""+file+"\"\n"+can.onexport.content(can)+"\n_can_name = \"\"\nconsole.log(\"once\")") - can.runAction(event, mdb.RENDER, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)], function(msg) { - can.onimport.profile(can, msg) - }) - }, - show: function(event, can) { can.request(event, {_toast: "渲染中..."}) - if (can.base.endWith(can.Option(nfs.FILE), ".js")) { - var file = can.base.Path("/require/", can.Option(nfs.PATH), can.Option(nfs.FILE)) - delete(Volcanos.meta.cache[file]), eval("\n_can_name = \""+file+"\"\n"+can.onexport.content(can)+"\n_can_name = \"\"\nconsole.log(\"once\")") - } - can.runAction(event, mdb.RENDER, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)], function(msg) { - can.onimport.profile(can, msg) - can.onappend._status(can, msg.Option(ice.MSG_STATUS), can.page.Append(can, can.ui._profile_output, ["status"]).first) - can.page.Select(can, can.ui._profile_output, "table.content", function(target) { - can.page.style(can, target, "max-height", "1000px") - }) - }) - }, - exec: function(event, can) { can.request(event, {_toast: "执行中...", "some": "run"}) - can.runAction(event, mdb.ENGINE, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)], function(msg) { - can.onimport.display(can, msg) - }) - }, - clear: function(event, can) { - if (can.page.Select(can, can._root._target, ".input.float", function(item) { return can.page.Remove(can, item) }).length > 0) { return } - if (can.page.Select(can, can._status, "legend.select", function(item) { return item.click(), item }).length > 0) { return } - - if (can.ui.display.style.display == "") { return can.onmotion.hidden(can, can.ui.display), can.onimport.layout(can) } - if (can.ui.profile.style.display == "") { return can.onmotion.hidden(can, can.ui.profile), can.onimport.layout(can) } - can.onmotion.toggle(can, can.ui.project), can.onimport.layout(can) - }, - back: function(event, can) { can.history.pop(); var last = can.history.pop() - last && can.onimport.tabview(can, last.path, last.file, last.line) - can.Status("跳转数", can.history.length) - }, - - scrollLine: function(can, count) { var size = 20; can.current.scroll(count*size) }, - appendLine: function(can, value) { - var ui = can.page.Append(can, can.ui.content, [{type: html.TR, list: [ - {view: ["line unselectable", html.TD, ++can.max], onclick: function(event) { - can.onaction.selectLine(event, can, ui.tr) - }, ondblclick: function(event) { - can.onaction.favorLine(can, ui.text.innerText) - }}, - - {view: [html.TEXT, html.TD], inner: can.onsyntax._parse(can, value), onclick: function(event) { - can.onaction.selectLine(event, can, ui.tr) - - }, ondblclick: function(event) { - var s = document.getSelection().toString(), str = ui.text.innerText - var begin = str.indexOf(s), end = begin+s.length - - for (var i = begin; i >= 0; i--) { - if (str[i].match(/[a-zA-Z0-9_.]/)) { s = str.slice(i, end) } else { break } - } - - if (s.indexOf("kit.") == 0) { s = s.replace("kit.", "toolkits.") } - if (s.indexOf(ice.PT) == 0) { s = s.slice(1) } - can.onaction.searchLine(event, can, s) - }} - ]}]); return ui.tr - }, - _getLine: function(can, line) { - return can.page.Select(can, can.ui.content, html.TR, function(item, index, array) { if (item == line || index+1 == line) { return item } })[0] - }, - _getLineno: function(can, line) { - return can.page.Select(can, can.ui.content, html.TR, function(item, index, array) { if (item == line || index+1 == line) { return index+1 } })[0] - }, - selectLine: function(event, can, line) { if (!line) { return parseInt(can.core.Value(can.page.Select(can, can.ui.content, [[[html.TR, html.SELECT], [html.TD, "line"]]])[0], "innerText")) } - can.page.Select(can, can.ui.content, html.TR, function(item, index, array) { if (line < 0 || line > array.length) { return } - if (!can.page.ClassList.set(can, item, html.SELECT, item == line || index+1 == line)) { return } - var ls = can.file.split(ice.PS) - if (ls.length > 4) { ls = [ls.slice(0, 2).join(ice.PS)+"/.../"+ls.slice(-2).join(ice.PS)] } - line = item, can.Status(kit.Dict("文件名", ls.join(ice.PS), "解析器", can.parse, "当前行", can.onexport.position(can, can.Option(nfs.LINE, index+1)))) - }) - - can.base.isObject(line) && can.page.Select(can, line, "td.text", function(item) { - can.current = { - window: function() { return parseInt(can.ui.content.offsetHeight/can.current.line.offsetHeight) }, - scroll: function(count) { if (count) { can.ui.content.scrollTop += count*can.current.line.offsetHeight } - return parseInt((can.current.line.offsetTop-can.ui.content.scrollTop)/can.current.line.offsetHeight) - }, - - prev: function() { return line.previousSibling }, - next: function() { return line.nextSibling }, - line: line, text: function(text) { - return text != undefined && can.onaction.modifyLine(can, line, text), item.innerText - }, - } - - if (!event) { - var scroll = can.current.scroll(); if (scroll < 5) { can.current.scroll(scroll-5) } else { - var window = can.current.window(); if (scroll > window/2) { can.current.scroll(scroll-window/2) } - } - } else { - var scroll = can.current.scroll(); if (scroll < 3) { can.current.scroll(scroll-3) } else { - var window = can.current.window(); if (scroll > window-4) { can.current.scroll(scroll-window+4) } - } - } - - var push = can.onimport.history(can, {path: can.Option(nfs.PATH), file: can.Option(nfs.FILE), line: can.Option(nfs.LINE), text: can.current.text()}) - if (can.isCmdMode()) { (location.hash = location.pathname.indexOf(can.Option(nfs.FILE)) > -1? "": push.file+ice.FS+(push.line||1)) } - can.onaction._selectLine(event, can) - }) - }, - _selectLine: function(event, can) { }, - searchLine: function(event, can, value) { - can.runAction(can.request(event, {text: value}, can.Option()), "navigate", [], function(msg) { - msg.Append(nfs.FILE)? can.onimport.tabview(can, msg.Append(nfs.PATH), msg.Append(nfs.FILE), msg.Append(nfs.LINE)): - can.user.toast(can, "not found") - }) - return - - if (can.ui.search) { - can.ui.search.Update(event, [ctx.ACTION, nfs.TAGS, value.trim()]) - } else { - can.runAction(event, nfs.TAGS, [value], function(msg) { - msg.Append(nfs.FILE)? can.onimport.tabview(can, msg.Append(nfs.PATH), msg.Append(nfs.FILE), msg.Append(nfs.LINE)): - can.user.toast(can, "not found") - }) - } - }, - favorLine: function(event, can) { }, }) -Volcanos(chat.ONEXPORT, {help: "导出数据", list: ["文件数", "解析器", "文件名", "当前行", "跳转数", "标签数"], +Volcanos(chat.ONEXPORT, {help: "导出数据", sess: function(can) { can.user.localStorage(can, "web.code.inner.sess", {"plug": can.onexport.plug(can), "exts": can.onexport.exts(can), "tabs": can.onexport.tabs(can)}) }, - keys: function(can, file, path) { return (path||can.Option(nfs.PATH))+ice.DF+(file||can.Option(nfs.FILE)) }, - tabs: function(can) { return can.core.Item(can.tabview, function(key, msg) { return key+ice.DF+msg.Option(nfs.LINE) }) }, + tabs: function(can) { return can.core.Item(can.tabview, function(key, msg) { return key+ice.DF+can.Option(nfs.LINE) }) }, plug: function(can) { return can.core.Item(can.toolkit) }, exts: function(can) { return can.core.Item(can.plugins) }, - position: function(can, index, total) { total = total||can.max; return (parseInt(index))+ice.PS+parseInt(total)+" = "+parseInt((index)*100/total)+"%" }, - content: function(can) { return can.page.Select(can, can.ui.content, "td.text", function(item) { return item.innerText }).join(ice.NL) }, }) diff --git a/plugin/local/code/inner/syntax.js b/plugin/local/code/inner/syntax.js index d612139f..51681abf 100644 --- a/plugin/local/code/inner/syntax.js +++ b/plugin/local/code/inner/syntax.js @@ -1,4 +1,18 @@ Volcanos(chat.ONSYNTAX, {help: "语法高亮", + makefile: { + prefix: { + "#": code.COMMENT, + }, + suffix: { + ":": code.COMMENT, + }, + keyword: { + "ifeq": code.KEYWORD, + "ifneq": code.KEYWORD, + "else": code.KEYWORD, + "endif": code.KEYWORD, + }, + }, h: { render: {}, link: "c", @@ -170,6 +184,7 @@ Volcanos(chat.ONSYNTAX, {help: "语法高亮", }, }, js: { + render: {}, split: { operator: "{[(.,:;!|<*>)]}", }, @@ -300,7 +315,11 @@ Volcanos(chat.ONSYNTAX, {help: "语法高亮", "fixed": "constant", }, }, + iml: { + render: {}, + }, zml: { + render: {}, prefix: { "# ": code.COMMENT, }, diff --git a/plugin/local/code/vimer.js b/plugin/local/code/vimer.js index 2a380db1..0fe586fa 100644 --- a/plugin/local/code/vimer.js +++ b/plugin/local/code/vimer.js @@ -1,10 +1,12 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, target) { can.require(["inner.js"], function(can) { can.onimport.inner_init(can, msg, function() { can.undo = [], can.redo = [] - can.onkeymap._build(can), can.onkeymap._plugin({}, can) - can.base.isFunc(cb) && cb(msg) - }, target) } , function(can, name, sub) { name == chat.ONIMPORT && (can.onimport.inner_init = sub._init) - can.page.ClassList.add(can, can._fields, "inner") - can.page.ClassList.add(can, can._fields, "vimer") + can.onengine.listen(can, "tabview.line.select", function(msg) { can.onaction._selectLine(msg._event, can) }) + can.core.Item(can.onkeymap._mode.plugin, function(key, value) { + can.onkeymap._mode.normal[key] = can.onkeymap._mode.normal[key]||value + }), can.onimport._input(can), can.onkeymap._build(can), can.onkeymap._plugin({}, can) + can.page.ClassList.add(can, can._fields, code.VIMER), can.base.isFunc(cb) && cb(msg) + can.onengine.plugin(can, can.onplugin) + }, target) }, function(can, name, sub) { name == chat.ONIMPORT && (can.onimport.inner_init = sub._init) if (name == chat.ONACTION) { can._trans = can.base.Copy(can._trans||{}, sub._trans) } if (name == chat.ONKEYMAP) { can.core.Item(sub._mode, function(mode, value) { var list = can.onkeymap._mode[mode] = can.onkeymap._mode[mode]||{} @@ -24,81 +26,74 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar }, onclick: function(event) { can.onkeymap._insert(event, can) - }}, {view: ["complete"]}, + }}, {view: [code.COMPLETE]}, ]); can.ui.current = ui.current, can.ui.complete = ui.complete }, }, [""]) -Volcanos(chat.ONFIGURE, {help: "导航索引", +Volcanos(chat.ONFIGURE, {help: "索引导航", + create: function(can, target, zone, path) { + if (can.isCmdMode()) { + can.onappend._action(can, can.base.Obj(can._msg.Option(ice.MSG_ACTION)), target) + } else { + can.onmotion.hidden(can, target.parentNode) + } + }, source: function(can, target, zone, path) { var total = 0 function show(path, target) { can.run(can.request({}, {dir_root: path, dir_deep: true}), [ice.PWD], function(msg) { var list = msg.Table() can.core.List(list, function(item) { if (can.Option(nfs.FILE).indexOf(item.path) == 0) { item.expand = true } - item._init = function(target) { target.onmouseenter = function(event) { can.user.carteRight(event, can, { - "trash": function() { can.onaction._run(event, can, nfs.TRASH, [can.base.Path(path, item.path)]) }, - _engine: function(event, can, button) { can.onaction[button](event, can, button) }, - }, ["trash"]) } } + item._menu = shy({trash: function(event) { can.onaction._run(event, can, nfs.TRASH, [can.base.Path(path, item.path)]) }}) }) - can.onimport.tree(can, list, nfs.PATH, ice.PS, function(event, item) { - can.onimport.tabview(can, path, item.path) // 显示文件 - }, target), can.Status("文件数", total += msg.Length()) - can.page.Modify(can, zone._search, {placeholder: "search in "+total+" item"}) - }, true) } if (path.length == 1) { return show(path[0], target) } + can.onimport.tree(can, list, nfs.PATH, ice.PS, function(event, item) { can.onimport.tabview(can, path, item.path) }, target) + can.Status("文件数", total += msg.Length()), zone._total(total) + }, true) } + if (path.length == 1) { return show(path[0], target) } + can.onmotion.delay(can, function() { can.onmotion.hidden(can, target.previousSibling) }) can.onimport.zone(can, can.core.List(path, function(path) { return {name: path, _init: function(target) { show(path, target) }} }), target) - can.onmotion.delay(can, function() { target.previousSibling.innerHTML = "" }) + }, + website: function(can, target, zone) { + can.run(can.request({}, {dir_root: "src/website/", dir_deep: true}), [ice.PWD], function(msg) { var list = msg.Table() + can.core.List(list, function(item) { if (can.Option(nfs.FILE).indexOf(item.path) == 0) { item.expand = true } }) + can.onimport.tree(can, list, nfs.PATH, ice.PS, function(event, item) { can.onimport.tabview(can, "src/website/", item.path) }, target) + zone._total(msg.Length()) + }, true) + }, + dream: function(can, target, zone) { var call = arguments.callee + can.runAction({}, ice.RUN, [web.DREAM], function(msg) { msg.Table(function(item) { var color = item.status == cli.START? "": "gray" + can.page.style(can, can.onimport.item(can, item, function(event) { + can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys(can.misc.Search(can, ice.POD), item.name), web.DREAM) + }, function(event) { + return shy({}, kit.Dict(cli.START, [cli.OPEN, cli.STOP], cli.STOP, [cli.START, nfs.TRASH])[item.status], function(event, button) { + can.runAction(can.request({}, item), ice.RUN, [web.DREAM, ctx.ACTION, button], function(msg) { + if (can.sup.onimport._process(can.sup, msg)) { return } + can.onmotion.clear(can, target), call(can, target, zone) + }) + }) + }, target), {color: color}) + }), zone._total(msg.Length()) }) + return shy(kit.Dict( + cli.REFRESH, function(event, can, button) { zone.refresh() }, + mdb.CREATE, function(event, can, button) { can.onaction.dream(event, can, web.DREAM) }, + code.PUBLISH, function(event, can, button) { can.runAction(event, button, [], function(msg) { can.user.toastConfirm(can, msg.Result(), button) }) }, + )) + }, + xterm: function(can, target, zone) { + can.runAction({}, ice.RUN, [code.XTERM], function(msg) { msg.Table(function(item) { + can.onimport.item(can, item, function(event) { + can.onimport.tabview(can, ctx.COMMAND, code.XTERM, item.hash) + }, function(event) {}, target) + }), zone._total(msg.Length()) }) }, plugin: function(can, target, zone) { var total = 0 can.onimport.tree(can, can.core.Item(can.onengine.plugin.meta, function(key) { return total++, {index: can.base.trimPrefix(key, "can.")} }), ctx.INDEX, ice.PT, function(event, item) { - can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys("can", item.index), ctx.INDEX) // 显示插件 - }, target) - can.page.Modify(can, zone._search, {placeholder: "search in "+total+" item"}) + can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys("can", item.index), ctx.INDEX) + }, target), zone._total(total) }, module: function(can, target, zone) { can.runAction(can.request({}, {fields: ctx.INDEX}), ctx.COMMAND, [mdb.SEARCH, ctx.COMMAND], function(msg) { - can.page.Modify(can, zone._search, {placeholder: "search in "+msg.Length()+" item"}) can.onimport.tree(can, msg.Table(), ctx.INDEX, ice.PT, function(event, item) { - can.onimport.tabview(can, can.Option(nfs.PATH), item.index, ctx.INDEX) // 显示模块 - }, target) - }) - }, - dream: function(can, target, zone) { - var call = arguments.callee - can.runAction({}, ice.RUN, [web.DREAM], function(msg) { - can.page.Modify(can, zone._search, {placeholder: "search in "+msg.Length()+" item"}) - msg.Table(function(item) { - function carte(event) { - var list = []; switch (item.status) { - case "start": list = ["open", "stop"]; break - case "stop": list = ["start", "trash"]; break - } - can.user.carteRight(event, can, {}, list, function(event, action) { - can.runAction(can.request({}, item), ice.RUN, [web.DREAM, ctx.ACTION, action], function(msg) { - can.onmotion.clear(can, target), call(target) - }) - }) - } - - var color = item.status == "start"? "": "gray" - can.page.Append(can, target, [{view: html.ITEM, list: [{text: [item.name, html.DIV], style: {color: color}, onmouseenter: carte}], onclick: function() { - can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys(can.misc.Search(can, "pod"), item.name), web.DREAM) // 显示空间 - }}]) - }) - }) - // _menu: shy("", { - // "create": function(event, can, button) { can.onaction.dream(event, can, "dream") }, - // "refresh": function(event, can, button) { can.ui.dreams.refresh() }, - // "publish": function(event, can, button) { can.runAction(event, button, [], function(msg) { can.user.toastConfirm(can, msg.Result(), button) }) }, - // }, ["create", "refresh", "publish"], function() {}) - }, - xterm: function(can, target, zone) { - can.runAction({}, ice.RUN, [code.XTERM], function(msg) { - can.page.Modify(can, zone._search, {placeholder: "search in "+msg.Length()+" item"}) - msg.Table(function(item) { - can.onimport.item(can, item, function(event) { - can.onimport.tabview(can, ctx.COMMAND, code.XTERM, item.hash) // 显示模块 - }, function(event) { - - }, target) - }) + can.onimport.tabview(can, can.Option(nfs.PATH), item.index, ctx.INDEX) + }, target), zone._total(msg.Length()) }) }, }) @@ -128,10 +123,10 @@ Volcanos(chat.ONKEYMAP, {help: "键盘交互", n: shy("命令模式", function(event, can) { can.onkeymap._normal(event, can) }), ":": shy("底行模式", function(event, can) { can.onimport.toolkit(can, {index: "cli.system"}, function(sub) { can.toolkit["cli.system"] = sub.select() }) }), - s: shy("保存文件", function(event, can) { can.onaction.save(event, can, "save") }), - d: shy("创建空间", function(event, can) { can.onaction.dream(event, can, "dream") }), - m: shy("添加模块", function(event, can) { can.onaction.autogen(event, can, "autogen") }), - c: shy("编译项目", function(event, can) { can.onaction.compile(event, can, "compile") }), + s: shy("保存文件", function(event, can) { can.onaction.save(event, can, nfs.SAVE) }), + d: shy("创建空间", function(event, can) { can.onaction.dream(event, can, web.DREAM) }), + m: shy("添加模块", function(event, can) { can.onaction.autogen(event, can, code.AUTOGEN) }), + c: shy("编译项目", function(event, can) { can.onaction.compile(event, can, code.COMPILE) }), }, normal_ctrl: { f: shy("向下翻页", function(event, can, target, count) { @@ -203,10 +198,6 @@ Volcanos(chat.ONKEYMAP, {help: "键盘交互", var line = can.onaction.insertLine(can, can._last_text, can.current.line) can.undo.push(function() { can.onaction.deleteLine(can, line), can.onaction.selectLine(event, can, line+1) }) }), - - s: shy("保存文件", function(event, can) { can.onaction.save(event, can, "save") }), - m: shy("添加模块", function(event, can) { can.onaction.autogen(event, can, "autogen") }), - c: shy("编译项目", function(event, can) { can.onaction.compile(event, can, "compile") }), }, insert: { Escape: shy("退出编辑", function(event, can) { if (event.key != "Escape") { return } @@ -249,8 +240,8 @@ Volcanos(chat.ONKEYMAP, {help: "键盘交互", }, }, _engine: {}, }) -Volcanos(chat.ONACTION, {help: "控件交互", list: [], - // list: [nfs.SAVE, code.COMPILE, code.AUTOGEN, nfs.SCRIPT, chat.WEBSITE, web.DREAM], +Volcanos(chat.ONACTION, {help: "控件交互", + _trans: {source: "源码", module: "模块", dreams: "空间", website: "网页"}, _run: function(event, can, button, args, cb) { can.runAction(event, button, args, cb||function(msg) { can.onimport.tabview(can, msg.Option(nfs.PATH)||can.Option(nfs.PATH), msg.Option(nfs.FILE)), can.ui.source.refresh() @@ -258,19 +249,17 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], }) }, _runs: function(event, can, button, cb) { var meta = can.Conf(); can.request(event, {action: button}) - can.user.input(event, can, meta.feature[button], function(args) { - can.onaction._run(event, can, button, args, cb) - }) + can.user.input(event, can, meta.feature[button], function(args) { can.onaction._run(event, can, button, args, cb) }) }, + save: function(event, can, button) { can.request(event, {file: can.Option(nfs.FILE), content: can.onexport.content(can)}) can.onaction._run(event, can, button, [can.parse, can.Option(nfs.FILE), can.Option(nfs.PATH)]) }, compile: function(event, can, button) { var toast = can.user.toastProcess(can, "编译中...") can.runAction(can.request(event), button, [], function(msg) { toast.close() if (msg.Length() == 0) { var toast1 = can.user.toastProcess(can, "重启中...") - can.onmotion.delay(can, function() { toast1.close(), can.onaction[cli.SHOW]({}, can) }, 5000) + can.onmotion.delay(can, function() { toast1.close(), can.onaction[cli.SHOW]({}, can) }, 3000) } else { can.ui.search._show(msg) } - }) }, autogen: function(event, can, button) { can.onaction._runs(event, can, button, function(msg) { @@ -284,11 +273,11 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], can.onaction._runs(can.request(event, {file: can.base.trimSuffix(can.Option(nfs.FILE), can.base.Ext(can.Option(nfs.FILE)))+nfs.JS}), can, button) }, website: function(event, can, button) { - can.onaction._runs(can.request(event, {path: "src/", file: (can.base.trimSuffix(can.Option(nfs.FILE), can.base.Ext(can.Option(nfs.FILE)))+nfs.ZML).split("/").pop()}), can, button) + can.onaction._runs(can.request(event, {path: "src/", file: (can.base.trimSuffix(can.Option(nfs.FILE), can.base.Ext(can.Option(nfs.FILE)))+nfs.ZML).split(ice.PS).pop()}), can, button) }, dream: function(event, can, button) { can.onaction._runs(can.request(event, {name: can.base.trimSuffix(can.Option(nfs.FILE).split(ice.PS).pop(), ice.PT+can.base.Ext(can.Option(nfs.FILE)))}), can, button, function(msg) { can.ui.dream.refresh() - can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys(can.misc.Search(can, "pod"), msg.Option(mdb.NAME)), web.DREAM) // 显示空间 + can.onimport.tabview(can, can.Option(nfs.PATH), can.core.Keys(can.misc.Search(can, ice.POD), msg.Option(mdb.NAME)), web.DREAM) can.user.toastSuccess(can) }) }, @@ -298,7 +287,12 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], can.user.toastSuccess(can) }) }, - "添加": function(event, can) { can.user.input(event, can, [ctx.INDEX], function(list) { can.onimport.tabview(can, can.Option(nfs.PATH), list[0], ctx.INDEX) }) }, + + "添加": function(event, can) { + can.user.input(event, can, [ctx.INDEX], function(list) { + can.onimport.tabview(can, can.Option(nfs.PATH), list[0], ctx.INDEX) + }) + }, "插件": function(event, can) { can.user.input(event, can, [ctx.INDEX], function(list) { var sub = can.toolkit[list[0]]; if (sub) { sub.select(); return } @@ -312,26 +306,11 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], }) }, - listTags: function(event, can, button) { var list = [] - can.core.Item(can.request(event), function(key, value) { if (key.indexOf("_") == 0) { return } - list.push({zone: "msg", type: typeof value, name: key, text: can.base.isObject(value)? "": (value+"").split(ice.NL)[0], - path: "usr/volcanos/", file: "lib/misc.js", line: 1, - }) - }) - can.core.List([can.base, can.core, can.misc, can.page, can.user, can.onengine, can.ondaemon, can.onappend, can.onlayout, can.onmotion, can.onkeymap], function(lib) { - can.core.Item(lib, function(key, value) { if (key.indexOf("_") == 0 || !lib.hasOwnProperty(key)) { return } - list.push({zone: lib._name, type: typeof value, name: key, text: can.base.isObject(value)? "": (value+"").split(ice.NL)[0], - path: "usr/volcanos/", file: lib._path, line: 1, - }) - }) - }) - can.runAction(can.request(event, {text: can.base.Format(list)}), button) - }, _complete: function(event, can, target) { target = target||can.ui.complete if (event == undefined) { return } - const PRE = "pre" var pre = can.ui.current.value.slice(0, can.ui.current.selectionStart) var end = can.ui.current.value.slice(can.ui.current.selectionStart) + const PRE = "pre" var type = can.core.Split(pre).pop()||"" var name = can.core.Split(type, "", ice.PT).pop()||"" @@ -340,7 +319,7 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], function update() { target._pre = pre, target._index = -1 can.current.line.appendChild(target), can.page.style(can, target, html.LEFT, can.ui.current.offsetLeft-1, html.MARGIN_TOP, can.ui.current.offsetHeight+4) - can.runAction(can.request(event, {text: pre}, can.Option()), "complete", [], function(msg) { + can.runAction(can.request(event, {text: pre}, can.Option()), code.COMPLETE, [], function(msg) { can.page.Appends(can, target, [{view: [PRE, html.DIV, pre]}]) can.onappend.table(can, msg, function(value, key, index) { return {text: [value, html.TD], onclick: function(event) { var left = can.ui.current.value.slice(0, can.ui.current.selectionStart)+value @@ -384,51 +363,12 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], case "": default: filter() - } - return - - function update0(target) { - can.request(event, {type: type, name: name, text: pre, file: can.Option(nfs.FILE)}) - can.runAction(event, "complete", [], function(msg) { target._msg = msg - if (msg.Length() == 0 && can.base.Ext(can.Option(nfs.FILE)) == nfs.JS) { - msg.name = [], msg.type = [], msg.text = [] - can.core.Item(can.core.Value(window, type), function(k, v) { - msg.Push(mdb.NAME, k), msg.Push(mdb.TYPE, typeof v) - try { msg.Push(mdb.TEXT, v) } catch (e) { msg.Push(mdb.TEXT, "") } - }) - msg.append = [mdb.NAME, mdb.TYPE, mdb.TEXT] - } - msg.Length() == 0 && pre.trim() == "" && can.core.Item(can.core.Value(can.onsyntax[can.parse], code.KEYWORD), function(k) { - msg.Push(mdb.NAME, k) - }) - - can.page.Appends(can, target, [{view: [PRE, html.DIV, pre]}]) - can.onappend.table(can, msg, function(value, key, index) { return {text: [value, html.TD], onclick: function(event) { - var left = can.ui.current.value.slice(0, can.ui.current.selectionStart)+value - can.current.text(can.ui.current.value = left+can.ui.current.value.slice(can.ui.current.selectionEnd)) - can.ui.current.focus(), can.ui.content.scrollLeft -= 10000, can.ui.current.setSelectionRange(left.length, left.length) - }} }, target) - }) - } - - can.current.line.appendChild(can.ui.complete), can.page.style(can, can.ui.complete, html.LEFT, can.ui.current.offsetLeft-1, html.MARGIN_TOP, can.ui.current.offsetHeight+4) - can.page.Select(can, target, "div.pre", function(item) { item.innerText = can.ui.complete._msg.Option(PRE)||pre }) - if (pre == "") { return update(target) } - - switch (event.key) { - case html.ENTER: update(target); break - case html.TAB: update(target); break - case ice.SP: update(target); break - case ice.PT: update(target); break - default: filter() || update(target); break - } can.ui.complete._index = -1 }, - _selectLine: function(event, can) { can.page.Select(can, can.current.line, "td.text", function(td) { can.ui.current.value = td.innerText can.current.line.appendChild(can.ui.current), can.page.style(can, can.ui.current, html.LEFT, td.offsetLeft-1, html.WIDTH, can.ui.content.style.width) - if (event) { + if (event && event.type) { if (event.type == "click" && can.mode != "insert") { can.onkeymap._insert(event, can) can.onmotion.delay(can, function() { can.onaction._complete(event, can) }) } @@ -437,6 +377,13 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], } }) }, + + _getLine: function(can, line) { + return can.page.Select(can, can.ui.content, html.TR, function(item, index, array) { if (item == line || index+1 == line) { return item } })[0] + }, + _getLineno: function(can, line) { + return can.page.Select(can, can.ui.content, html.TR, function(item, index, array) { if (item == line || index+1 == line) { return index+1 } })[0] + }, rerankLine: function(can, value) { can.max = 0 can.page.Select(can, can.ui.content, html.TR, function(item, index) { can.max++, can.page.Select(can, item, "td.line", function(item) { item.innerText = index+1 }) @@ -458,13 +405,25 @@ Volcanos(chat.ONACTION, {help: "控件交互", list: [], }) }) }, - favorLine: function(can, value) { - can.user.input(event, can, [{name: "zone", value: "hi"}, {name: "name", value: "hello"}], function(data) { - can.runAction(event, code.FAVOR, [ctx.ACTION, mdb.INSERT, mdb.ZONE, data.zone||"", - mdb.TYPE, can.parse, mdb.NAME, data.name||"", mdb.TEXT, (value||"").trimRight(), - nfs.PATH, can.Option(nfs.PATH), nfs.FILE, can.Option(nfs.FILE), nfs.LINE, can.Option(nfs.LINE), - ]) - }) - }, }) Volcanos(chat.ONEXPORT, {help: "导出数据", list: ["文件数", "模式", "按键", "解析器", "文件名", "当前行", "跳转数"]}) +Volcanos(chat.ONPLUGIN, {help: "注册插件", + "can.code.vimer.keymap": shy("按键", {}, ["mode", "key", ice.LIST, ice.BACK], function(can, msg, cmds) { + can.core.Item(can.onkeymap._mode, function(mode, value) { + (!cmds[0] || cmds[0] == mode) && can.core.Item(value, function(key, func) { + if (cmds[0] == mode && cmds[1] == key) { + msg.Push("key", "mode") + msg.Push("value", mode) + msg.Push("key", "key") + msg.Push("value", key) + msg.Push("key", "help") + msg.Push("value", func.help) + msg.Push("key", "func") + msg.Push("value", func.toString()) + } else if (!cmds[0] || !cmds[1]) { + msg.Push(kit.Dict("mode", mode, "key", key, "help", func.help||func.toString())) + } + }) + }), msg.StatusTimeCount() + }), +}) diff --git a/plugin/local/code/xterm.js b/plugin/local/code/xterm.js index 43666c6a..d80d584f 100644 --- a/plugin/local/code/xterm.js +++ b/plugin/local/code/xterm.js @@ -4,7 +4,7 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg) { can.o item.init && can.onmotion.delay(can, function() { can.onimport._input(can, item.init+ice.NL) }) can.onimport.layout(can), can.onappend._status(can), can.onappend.tools(can, msg, function(sub) { sub._item_click = function(value, key) { can.onimport._input(can, value+ice.NL) } - }), can.isCmdMode() && can.misc.Search(can, mdb.HASH) && can.sup.onaction.full({}, can) + }), can.isCmdMode() && can.misc.Search(can, mdb.HASH) && can.sup.onaction.full({}, can.sup) can.onimport._connect(can, item) }) }, @@ -33,7 +33,7 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg) { can.o } else { can.page.style(can, can._output, html.HEIGHT, can.ConfHeight(), html.WIDTH, can.ConfWidth(), html.MAX_WIDTH, "") } - can.onmotion.delay(can, function() { can.page.style(can, can._output, html.HEIGHT, "", html.WIDTH, "") }) + can.onmotion.delay(can, function() { can.page.style(can, can._output, html.HEIGHT, "", html.WIDTH, "") }, 500) can._current && can._current._fit.fit() }, grow: function(can, msg) { can._current.write(msg.Option(mdb.TEXT)) }, diff --git a/plugin/local/team/plan.js b/plugin/local/team/plan.js index 808e3509..dfe0a2dc 100644 --- a/plugin/local/team/plan.js +++ b/plugin/local/team/plan.js @@ -56,7 +56,7 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar var table = can.page.Appends(can, can.ui.profile, [{view: [chat.CONTENT, html.TABLE], list: [{th: [mdb.KEY, mdb.VALUE]}]}]).first can.core.Item(task, function(key, value) { key != "_target" && can.page.Append(can, table, [{ td: [key, key == ice.POD && value != ""? can.page.Format(html.A, can.misc.MergeURL(can, {pod: value}), value): value], - onclick: function(event) { if (can.page.tagis(html.INPUT, event.target) && event.target.type == html.BUTTON) { + onclick: function(event) { if (can.page.tagis(event.target, html.INPUT) && event.target.type == html.BUTTON) { can.run(can.request(event, task), [ctx.ACTION, event.target.name], function(msg) { can.Update() }) } }, ondblclick: function(event) { var msg = can.request() diff --git a/plugin/local/wiki/draw.js b/plugin/local/wiki/draw.js index 5bc47ce8..fe8df587 100644 --- a/plugin/local/wiki/draw.js +++ b/plugin/local/wiki/draw.js @@ -24,7 +24,7 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar can.svg = can.group = can.onimport._block(can, svg), can.onimport._group(can, svg).click() can.core.ItemCB(can.onaction, function(key, cb) { svg[key] = function(event) { cb(event, can) } }) can.page.Select(can, svg, mdb.FOREACH, function(item, index) { can.onimport._block(can, item) - can.page.tagis(svg.G, item) && item.Value(html.CLASS) && can.onimport._group(can, item) + can.page.tagis(item, svg.G) && item.Value(html.CLASS) && can.onimport._group(can, item) }) }) }, @@ -314,7 +314,7 @@ Volcanos(chat.ONACTION, {help: "组件菜单", list: [ resize: function(event, can, point, target) { target = target||event.target if (event.type == html.CLICK) { if (point.length == 1) { - can.current = {target: target, begin: can.core.List([target], function(item) { if (can.page.tagis(svg.G, item)) { return } + can.current = {target: target, begin: can.core.List([target], function(item) { if (can.page.tagis(item, svg.G)) { return } return { height: item.Val(html.HEIGHT), width: item.Val(html.WIDTH), x: item.Val(svg.X), y: item.Val(svg.Y), target: item, ship: can.core.List(item.Value(ice.SHIP), function(ship) { diff --git a/plugin/local/wiki/word.js b/plugin/local/wiki/word.js index 89d8e47e..82beb7e1 100644 --- a/plugin/local/wiki/word.js +++ b/plugin/local/wiki/word.js @@ -125,7 +125,7 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar can.runAction(event, chat.STORY, can.misc.concat(can, [data.type, data.name, data.text], cmds), cb) } - sub.ConfHeight(can.ConfHeight()), sub.ConfWidth(item.width = (width||can.ConfWidth())-(can.user.isWindows? 40: 20)) + sub.ConfHeight(can.base.Min(300, can.ConfHeight()-300)), sub.ConfWidth(item.width = (width||can.ConfWidth())-(can.user.isWindows? 40: 20)) can.page.style(can, sub._output, html.MAX_WIDTH, sub.ConfWidth()) can.core.Value(item, "auto.cmd") && can.onmotion.delay(function() { @@ -155,7 +155,7 @@ Volcanos(chat.ONACTION, {help: "控件交互", _trans: {view: "视图"}, play: function(event, can) { var list = [], current = [] can.page.Select(can, can._output, wiki.ITEM, function(item) { - can.page.tagis(["h1", "h2", "h3"], item) && list.push(current = []), current.push(item) + can.page.tagis(item, "h1", "h2", "h3") && list.push(current = []), current.push(item) }) can.onappend._init(can, {type: "story word float"}, [], function(sub) { diff --git a/plugin/state.js b/plugin/state.js index 783442d1..7249000c 100644 --- a/plugin/state.js +++ b/plugin/state.js @@ -191,11 +191,12 @@ Volcanos(chat.ONACTION, {help: "交互操作", list: [ }) }, - full: function(event, can) { + full: function(event, can) { var sub = can.core.Value(can, chat._OUTPUTS_CURRENT) if (can.isCmdMode()) { can.onmotion.hidden(can, can._legend), can.onmotion.hidden(can, can._option), can.onmotion.hidden(can, can._action), can.onmotion.hidden(can, can._status) can.ConfHeight(window.innerHeight), can.ConfWidth(window.innerWidth) - can.onimport.layout(can) + sub.ConfHeight(window.innerHeight), sub.ConfWidth(window.innerWidth) + sub.onimport.layout(sub) } else { can.onaction["切换全屏"](event, can) } diff --git a/plugin/story/parse.js b/plugin/story/parse.js index 89bb89cc..0ba5f97a 100644 --- a/plugin/story/parse.js +++ b/plugin/story/parse.js @@ -41,7 +41,7 @@ Volcanos(chat.ONAPPEND, {help: "渲染引擎", list: ["{", "}", html.HEAD, html. if (item.parentNode == target.parentNode) { height -= item.offsetHeight // 高度 } - }), can.page.style(can, target, html.HEIGHT, height) + }), can.page.style(can, target, html.HEIGHT, height-40) }) } break diff --git a/plugin/table.js b/plugin/table.js index 72ceb603..d87830d1 100644 --- a/plugin/table.js +++ b/plugin/table.js @@ -16,6 +16,17 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar _title: function(can, title) { can.user.title(title+ice.SP+(can.misc.Search(can, ice.POD)||location.host)) }, + _panel: function(can, target, action) { + var ui = can.page.Append(can, target, [html.ACTION, html.OUTPUT]) + var action = can.onappend._action(can, [], ui.action, kit.Dict( + cli.CLOSE, function(event) { can.onmotion.hidden(can, target), can.onimport.layout(can) }, + cli.CLEAR, function(event) { can.onmotion.clear(can, ui.output) }, + cli.SHOW, function(event) { can.onaction[cli.SHOW](event, can) }, + action, + mdb.PLUGIN, function(event) { can.user.input(event, can, [ctx.INDEX, ctx.ARGS], function(data) { can.onimport.plug(can, data, ui.output) }) }, + )); target._toggle = function(event, show) { action[show? cli.SHOW: cli.CLOSE](event) } + return ui + }, zone: function(can, list, target) { var color = [""] return can.page.Append(can, target, can.core.List(list, function(zone, index) { can.base.isString(zone) && (zone = {name: zone}); return zone && {view: html.ZONE+" "+zone.name, list: [ {view: html.NAME, inner: can.user.trans(can, zone.name), style: {background: color[index%color.length]}, onclick: function() { @@ -43,22 +54,22 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar }}], target, {}) }}, {view: html.LIST, _init: function(target) { can.ui[zone.name] = zone + zone._total = function(total) { can.page.Modify(can, zone._search, {placeholder: "search in "+total+" item"}) } zone._target = target, zone.refresh = function() { can.onmotion.clear(can, target), zone._init(target, zone) } - can.base.isFunc(zone._init) && zone._init(target, zone) + can.base.isFunc(zone._init) && (zone._menu = zone._init(target, zone)||zone._menu) }} ]} })) }, tree: function(can, list, field, split, cb, target, node) { node = node||{"": target}; can.core.List(list, function(item) { item[field] && can.core.List(item[field].split(split), function(value, index, array) { if (!value) { return } - var last = array.slice(0, index).join(split), name = array.slice(0, index+1).join(split) - if (node[name]) { return } - var ui = can.page.Append(can, node[last], [{view: "item", list: [{view: ["switch", "div", (index==array.length-1?"":"⌃")]}, {view: ["name", html.DIV, value+(index==array.length-1?"":"")], _init: item._init}], onclick: function(event) { - index < array.length - 1? can.onmotion.toggle(can, node[name], function() { - can.page.ClassList.add(can, ui["switch"], "open") - }, function() { - can.page.ClassList.del(can, ui["switch"], "open") - }): can.base.isFunc(cb) && cb(event, item) + var last = array.slice(0, index).join(split), name = array.slice(0, index+1).join(split); if (node[name]) { return } + var ui = can.page.Append(can, node[last], [{view: "item", list: [{view: ["switch", "div", (index==array.length-1?"":"⌃")]}, {view: ["name", html.DIV, value+(index==array.length-1?"":"")], _init: item._init, onmouseenter: function(event) { if (!item._menu) { return } + can.user.carteRight(event, can, item._menu.meta, item._menu.list||can.core.Item(item._meta.meta), function(event, button) { + (item._menu.meta[button]||item._menu)(event, can, button) + }) + }}], onclick: function(event) { + index < array.length - 1? can.onmotion.toggle(can, node[name], function() { can.page.ClassList.add(can, ui["switch"], "open") }, function() { can.page.ClassList.del(can, ui["switch"], "open") }): can.base.isFunc(cb) && cb(event, item) if (node[name].childElementCount == 2) { node[name].firstChild.click() } }}, {view: html.LIST, style: {display: html.NONE}, _init: function(list) { item.expand && can.page.style(can, list, html.DISPLAY, html.BLOCK) }}]) node[name] = ui.list @@ -77,7 +88,14 @@ Volcanos(chat.ONIMPORT, {help: "导入数据", _init: function(can, msg, cb, tar var ui = can.page.Append(can, target, [{view: [html.ITEM, html.DIV, item.nick||item.name], onclick: function(event) { cb(event, ui.first, event.target._list && can.onmotion.toggle(can, event.target._list)) can.onmotion.select(can, target, can.core.Keys(html.DIV, html.ITEM), ui.first) - }, onmouseenter: function(event) { can.base.isFunc(cbs) && cbs(event, ui.first) }, + }, onmouseenter: function(event) { + if (can.base.isFunc(cbs)) { + var menu = cbs(event, ui.first) + if (menu) { + can.user.carteRight(event, can, menu.meta, menu.list, menu) + } + } + }, }]); return ui.first }, itemlist: function(can, list, cb, cbs, target) { diff --git a/proto.js b/proto.js index c16eff9e..c9a78b53 100644 --- a/proto.js +++ b/proto.js @@ -141,6 +141,10 @@ var code = { CONSTANT: "constant", DATATYPE: "datatype", FUNCTION: "function", + + TEMPLATE: "template", + COMPLETE: "complete", + NAVIGAET: "navigate", } var wiki = { TITLE: "title", BRIEF: "brief", REFER: "refer", SPARK: "spark", @@ -229,6 +233,7 @@ var html = { // FIELDSET FIELDSET: "fieldset", LEGEND: "legend", OPTION: "option", ACTION: "action", OUTPUT: "output", STATUS: "status", FORM_OPTION: "form.option", DIV_ACTION: "div.action", DIV_OUTPUT: "div.output", DIV_STATUS: "div.status", + DIV_CONTENT: "div.content", FIELDSET_PANEL: "fieldset.panel", FIELDSET_PLUGIN: "fieldset.plugin", FIELDSET_STORY: "fieldset.story", FIELDSET_HEAD: "fieldset.head", FIELDSET_FOOT: "fieldset.foot", FIELDSET_LEFT: "fieldset.left", FIELDSET_MAIN: "fieldset.main",