Volcanos("onimport", {help: "导入数据", _init: function(can, msg, list, cb, target) { var list = []; can.onengine.listen(can, "resize", function(width, height) { can.Conf({width: width, height: height}), can.core.Delay(list, 100, function() { can.onimport._init(can, msg, list, cb, target) }) }) can.page.Modify(can, can._target, {style: {"max-width": can.Conf("width")}}) var width = can.Conf("width"), height = can.Conf("height") can.ui = can.page.Appends(can, target, [ {type: "table", list: [{type: "tr", list: [ {type: "td", list: [{view: "project", style: {"max-height": height-240, display: "none"}} ]}, {type: "td", list: [{view: "profile", style: {"max-height": height-240}, list: [ {view: ["content", "table"]}, ]}], style: {"min-width": width-60, "max-width": width-30}}, ]}, ]}, {view: "search", style: {display: "none"}, list: [{view: "action", list: [ {input: ["word", function(event) { if (event.key == "Enter") { can.onaction.searchLine(event, can, can.ui.word.value) } }], value: "main", onfocus: function(event) { event.target.setSelectionRange(0, -1) }}, {button: ["搜索", function(event) { can.onaction.searchLine(event, can, can.ui.word.value) }]}, {button: ["返回", function(event) { var last = can.history.pop(); last = can.history.pop() last && can.onimport.tabview(can, last.path, last.file, last.line) }]}, {button: ["关闭", function(event) { can.page.Modify(can, can.ui.search, {style: {display: "none"}}) }]}, ]}, {view: "tags", style: {"max-height": 160}}, ]}, ]) typeof cb == "function" && cb(msg) can.history = can.history || [] can.tabview = {}, can.tabview[can.Option("path")+can.Option("file")] = msg msg.Option({path: can.Option("path"), file: can.Option("file"), line: can.Option("line")||1}) can.onimport.tabview(can, can.Option("path"), can.Option("file"), can.Option("line")||1) can.onimport.project(can, can.Option("path")) }, tabview: function(can, path, file, line, cb) { function show() { can._msg && can._msg.Option("line", can.Option("line")) can._msg = can.tabview[path+file] can.Option({path: path, file: file, line: line||parseInt(can._msg.Option("line"))||1}) can.onsyntax._init(can, can._msg) typeof cb == "function" && cb() } if (can.tabview[path+file]) { return show() } can.run({}, ["action", "render", can.base.Ext(file||path), file, path], function(msg) { msg.Option({path: path, file: file, line: line||1}) can.tabview[path+file] = msg can.page.EnableDrop(can, can._action, "div.file", can.page.Append(can, can._action, [ {view: ["file", "div", file.split("/").pop()], title: file, onclick: function(event) { can.onimport.tabview(can, path, file, "", cb) }} ]).first).click() }, true) }, project: function(can, path, cb) { can.Option({path: path}) var msg = can.request({}, {dir_root: path, dir_deep: true}) can.run(msg._event, ["action", "dir", "./"], function(msg) { can.ui.project.innerHTML = "" msg.path && can.Status("文件数", msg.path.length) can.onappend.tree(can, msg, "path", "/", can.ui.project, function(event, value) { can.onimport.tabview(can, can.Option("path"), value.path) }), typeof cb == "function" && cb() }, true) }, }, ["/plugin/local/code/inner.css"]) Volcanos("onsyntax", {help: "语法高亮", list: ["keyword", "prefix", "line"], _init: function(can, msg) { can._msg = msg can.page.Select(can, can._action, "div.file", function(item) { can.page.ClassList.set(can, item, "select", item.title == msg.Option("file")) }) // caches save can.core.List(["content"], function(item) { can.page.Cache(can.file+item, can.ui[item], { scrollTop: can.ui.profile.parentNode.scrollTop, current: can.current, max: can.max, }) }) can.file = can.base.Path(msg.Option("path"), msg.Option("file")) can.parse = can.base.Ext(can.file) can.Status("模式", "normal") // caches load var cache = false; can.core.List(["content"], function(item) { var p = can.page.Cache(can.file+item, can.ui[item]); if (p != undefined) { cache = true can.ui.profile.parentNode.scrollTo(0, p.scrollTop) can.onaction.selectLine(can, p.current.line) can.max = p.max } }); if (cache) { return } function init(p) { can.max = 0, can.core.List(can.ls = msg.Result().split("\n"), function(item) { can.onaction.appendLine(can, item) }), can.onaction.selectLine(can, can.Option("line")||1) } // plugin var p = can.onsyntax[can.parse]; !p? can.run({}, ["action", "plugin", can.parse, can.Option("file"), can.Option("path")], function(msg) { init(p = can.onsyntax[can.parse] = can.base.Obj(msg.Result())) }, true): init(p) }, _parse: function(can, line) { line = line || "", line = line.replace("<", "<").replace(">", ">") var p = can.onsyntax[can.parse]; if (!p) { return } p = can.onsyntax[p.link] || p function wrap(type, str) { return type? ''+str+'': str } p.keyword && (line = can.core.List(can.core.Split(line, p.split && p.split.space || " ", p.split && p.split.operator || "{[(|)]}"), function(item, index, array) { item = typeof item == "object"? item: {text: item}, p.word && (item = p.word(item, index, array)) var text = item.text; var key = item.keyword||p.keyword[text] switch (item.type) { case "space": return text case "string": return wrap("string", item.left+text+item.right) default: return wrap(key, text) } }).join("")) p.prefix && can.core.Item(p.prefix, function(pre, type) { if (line.trim().startsWith(pre)) { line = wrap(type, line) } }) p.suffix && can.core.Item(p.suffix, function(pre, type) { if (line.endsWith(pre)) { line = wrap(type, line) } }) return p.line? p.line(can, line): line }, }) Volcanos("onaction", {help: "控件交互", list: [], "返回": function(event, can) { var last = can.history.pop(); last = can.history.pop() last && can.onimport.tabview(can, last.path, last.file, last.line) can.Status("跳转数", can.history.length) }, project: function(event, can) { can.page.Toggle(can, can.ui.project, function() { // can.onimport.project(can, can.Option("path")) }) }, search: function(event, can) { can.page.Toggle(can, can.ui.search, function() { // can.onaction.searchLine(event, can, "") }) }, appendLine: function(can, value) { var index = ++can.max var ui = can.page.Append(can, can.ui.content, [{type: "tr", list: [{view: ["line", "td", index], onclick: function(event) { can.onaction.selectLine(can, ui.tr) }, ondblclick: function(event) { can.user.input(event, can, [ {_input: "text", name: "topic", value: "@key"}, {_input: "text", name: "name", value: "@key"}, ], function(event, button, meta, list) { can.run(event, [ "favor", "topic", meta.topic||"some", "type", can.parse, "name", meta.name||"some", "text", value, "path", can.Option("path"), "file", can.Option("file"), "line", can.Option("line"), ], function(msg) { can.user.toast(can, "收藏成功") }, true) return true }) }}, {view: ["text", "td"], _init: function(td) { var p = can.onsyntax._parse(can, value) typeof p == "object"? can.page.Appends(can, td, [p]): td.innerHTML = p }, onclick: function(event) { can.onkeymap && can.onkeymap._mode(can, "insert") can.onaction.selectLine(can, ui.tr) can.onkeymap && can.ui.editor.focus() // can.ui.editor.setSelectionRange(0, 0) // can.ui.editor.setSelectionRange(event.offsetX/10, event.offsetX/10) }, ondblclick: function(event) { var str = ui.text.innerText var s = document.getSelection().toString() var begin = str.indexOf(s) var 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(".") == 0) { s = s.slice(1) } if (s.indexOf("kit.") == 0) { s = s.replace("kit.", "toolkits.") } can.onaction.searchLine(event, can, s) }} ] }]) return ui.tr }, rerankLine: function(can, value) { can.max = 0 can.page.Select(can, can.ui.content, "tr", function(item, index) { can.max++, can.page.Select(can, item, "td.line", function(item) { item.innerText = index+1 }) }) }, modifyLine: function(can, line, value) { can.page.Select(can, can.ui.content, "tr", function(item, index) { if (item != line && index+1 != line) { return } can.page.Select(can, item, "td.text", function(item) { var p = can.onsyntax._parse(can, value) typeof p == "object"? can.page.Appends(can, item, [p]): item.innerHTML = p }) }) }, selectLine: function(can, line) { if (!line) { return } can.page.Select(can, can.ui.content, "tr", function(item, index) { can.page.ClassList.del(can, item, "select") if (item != line && index+1 != line) { return } can.page.ClassList.add(can, item, "select") line = item can.Option("line", index+1) can.Status("文件名", can.file) can.Status("解析器", can.parse) can.Status("当前行", can.onexport.position(can, index+1)) }) typeof line == "object" && can.page.Select(can, line, "td.text", function(item) { can.current = { scroll: function(x, y) { return can.ui.profile.parentNode.scrollBy(x, y) }, offset: function() { return can.ui.profile.parentNode.scrollTop }, window: function() { return can.ui.profile.parentNode.offsetHeight }, height: function() { return line.offsetHeight }, prev: function() { return line.previousSibling }, next: function() { return line.nextSibling }, line: line, text: function(text) { text != undefined && can.onaction.modifyLine(can, line, text) return item.innerText }, } var push = {path: can.Option("path"), file: can.Option("file"), line: can.Option("line")} can.core.Eq({path: push.path, file: push.file}, can.history[can.history.length-1]) || can.history.push(push) can.Status("跳转数", can.history.length) can.onkeymap && can.onkeymap.selectLine(can, line, item) }) }, searchLine: function(event, can, value) { value = value.trim() can.page.Modify(can, can.ui.search, {style: {display: ""}}) value = can.ui.word.value = value || can.ui.word.value || "main" var toast = can.user.toast(can, "搜索中...", value, 1000000) can.run(event, ["action", "search", can.parse, value, can.Option("path")], function(msg) { toast.Close() can.ui.tags.innerHTML = "" can.onappend.table(can, msg, can.ui.tags, "content", function(value, key, index, line) { can.Status("标签数", index+1) value = value.replace("<", "<").replace(">", ">"), value = value.replace("./", "") return {text: ["", "td"], list: [{text: [value, "div"]}], onclick: function(event) { line.line && can.onimport.tabview(can, can.Option("path"), line.file.replace("./", ""), parseInt(line.line), function() { var pos = can.current.offset()-can.current.line.offsetTop can.current.scroll(0, -pos) }) }} }) }, true) }, }) Volcanos("onexport", {help: "导出数据", list: ["文件数", "解析器", "文件名", "当前行", "跳转数", "标签数"], position: function(can, index, total) { total = total || can.max return (parseInt(index))+"/"+parseInt(total)+" = "+parseInt((index)*100/total)+"%" }, content: function(can) { return can.page.Select(can, can._output, "table.content td.text", function(item) { return item.innerText }).join("\n")+"\n" }, })