diff --git a/js/kunai.js b/js/kunai.js index a9d08eb..d6f7bfd 100644 --- a/js/kunai.js +++ b/js/kunai.js @@ -30,6 +30,7 @@ class Kunai { })) ) + try { this.log.disableBacktrace(true) this.log.info(`version ${KUNAI_PACKAGE.version} (https://github.com/cpprefjp/kunai/tree/v${KUNAI_PACKAGE.version})`) @@ -38,16 +39,13 @@ class Kunai { this.log.disableBacktrace(false) } - if (this.opts.compat) { - this.compat = new Compat(this.log) - } - this.ui = { navbar: null, sidebar: null, content: null, } this.initUI() + this.getSidebar = this.initSidebar() this.wand = new Wand(this.log) @@ -56,15 +54,21 @@ class Kunai { async cpprefjp() { this.load_impl(['cpprefjp', 'site']) - this.crs = await this.initCRSearch(true) + this.afterCR = this.initCRSearch(true) + this.crs = await this.afterCR } async boostjp() { this.load_impl(['boostjp', 'site']) - this.crs = await this.initCRSearch(false) + this.afterCR = this.initCRSearch(false) + this.crs = await this.afterCR } - load_impl(config) { + async load_impl(config) { + if (this.opts.compat) { + this.compat = new Compat(this.log, config) + } + const desc = config.join('/') this.log.info(`loading (${desc})`) $('body').addClass('kunai') @@ -114,13 +118,18 @@ class Kunai { this.ui.content = new UI.Content(l) } - async initCRSearch(isEnabled, onDatabase) { + async onDatabase(db) { + await this.ui.sidebar.onDatabase(db) + await this.ui.sidebar.treeview.onPageID(this.meta.page_id) + } + + async initCRSearch(isEnabled) { if (!isEnabled) return null - await this.initSidebar() + await this.getSidebar let crs = new CRSearch({ - onDatabase: ::this.ui.sidebar.onDatabase, + onDatabase: ::this.onDatabase, }) crs.database('https://cpprefjp.github.io/static/crsearch/crsearch.json') diff --git a/js/kunai/compat.js b/js/kunai/compat.js index 1a26fac..2872d99 100644 --- a/js/kunai/compat.js +++ b/js/kunai/compat.js @@ -1,6 +1,7 @@ class Compat { - constructor(log) { + constructor(log, repo) { this.log = log.makeContext('Compat') + this.repo = [].concat(repo) this.log.info('applying...') @@ -27,6 +28,8 @@ class Compat { } } + // $('#navbar-collapse ul.nav > li:nth-child(2) > a').text(repo.join('/')) + this.log.info('applied.') } // constructor diff --git a/js/kunai/meta/meta.js b/js/kunai/meta/meta.js index 8e94380..4d3f0b8 100644 --- a/js/kunai/meta/meta.js +++ b/js/kunai/meta/meta.js @@ -6,6 +6,8 @@ import * as Code from '../code' import {Logger} from 'nagato' +import URL from 'url-parse' + let Marked = require('marked') const MarkedOpts = { @@ -56,6 +58,12 @@ class Meta { this.log.info(`original_url: ${this.original_url}`, this.original_url) this.filename = [...this.original_url.pathname.split('/')].pop() + { + let ids = this.original_url.pathname.split('/').slice(5) + ids[ids.length - 1] = ids[ids.length - 1].replace(/\.md$/, '') + this.page_id = ids + } + this.raw_url = new URL(this.original_url) this.raw_url.host = 'raw.githubusercontent.com' this.raw_url.pathname = this.raw_url.pathname.replace( diff --git a/js/kunai/ui/sidebar.js b/js/kunai/ui/sidebar.js index 68d4b86..af42dcc 100644 --- a/js/kunai/ui/sidebar.js +++ b/js/kunai/ui/sidebar.js @@ -33,38 +33,24 @@ class Sidebar { this.e.addClass('legacy') } - this.initComponents() + this.getTreeview = this.initTreeview() } async onDatabase(db) { try { this.log.info(`onDatabase`, db) - this.tree = db.getTree(this.kc) - - // workaround name - for (let t of this.tree) { - if (t.root) { - const name = t.root.id.join() - const real_name = t.category.name - if (name !== real_name) { - this.log.warn(`got incorrect title '${name}'; expected = '${real_name}'. ignoring...`) - } - } - } + this.db = db - this.treeView.onData(this.tree) + let t = await this.getTreeview + await t.onData(db) } finally { this.e.removeClass('loading') } } - async initComponents() { - await this.initDynamicComponents() - } - - async initDynamicComponents() { + async initTreeview() { let e = null if (this.legacy) { @@ -73,7 +59,7 @@ class Sidebar { e = $('
').addClass('kunai-tree').addClass('v2').appendTo(this.e) } - this.treeView = new Treeview( + this.treeview = new Treeview( this.log, this.kc, this.e, @@ -81,6 +67,7 @@ class Sidebar { legacy: this.legacy } ) + return this.treeview } } diff --git a/js/kunai/ui/treeview.js b/js/kunai/ui/treeview.js index f69dcb0..db9e3fc 100644 --- a/js/kunai/ui/treeview.js +++ b/js/kunai/ui/treeview.js @@ -24,6 +24,8 @@ class DOM { this.lastBranchID = 0 this.branchPrevs = new Map + + this.expandElems = new WeakMap } static scrollEps = 8 @@ -82,22 +84,23 @@ class DOM { } } - async createContent(e, elem, obj) { - // this.log.debug(`createContent '${obj.self.id}'`, e, elem, obj) + async createContent(obj) { + this.log.debug(`createContent '${obj.self.id}'`, obj) switch (obj.self.id.type) { case IType.header: - return await this.createHeaderContent(e, elem, obj) + return await this.createHeaderContent(obj) default: - this.log.error('createContent', e, elem, obj) + this.log.error('createContent', obj) throw new Error(`unhandled index type in createContent`) } } - async createHeaderContent(e, elem, h) { + async createHeaderContent(h) { // this.log.debug(`createHeaderContent (${h.self.id.join()})`, e, elem, h) let empty = true + let elem = this.expandElems.get(h.self.id) if (h.classes && h.classes.length) { empty = false @@ -126,10 +129,23 @@ class DOM { return true // this.lazyLoaders.get(h.self.id) } - async doExpand(id, e) { - // this.log.debug(`doExpand '${id.join()}'`, id, e) - await this.lazyLoaders.get(id)(e) - $(e.target).closest('li').toggleClass('expanded') + async doExpand(id) { + // this.log.debug(`doExpand '${id.join()}'`, id) + await this.lazyLoaders.get(id)() + this.expandElems.get(id).toggleClass('expanded') + } + + async scrollAt(id) { + this.log.debug(`scrollAt '${id.join()}'`, id) + + const e = this.expandElems.get(id) + + let broot = e.closest('.kunai-branch') + // this.log.debug(`branch root`, broot) + + broot.animate({ + scrollTop: e.offset().top + }, 1) } async kunaiBranch(me, branchFor, scrollHandler) { @@ -226,18 +242,18 @@ class DOM { } async makeExpandable(elem, obj) { + // this.log.debug(`makeExpandable '${obj.self.id.join()}'`, elem, obj) + this.expandElems.set(obj.self.id, elem) this.lazyLoaders.set( obj.self.id, - async (e) => { - return await ::this.createContent(e, elem, obj) - } + async () => { await this.createContent(obj) } ) let bar = $('
', {class: 'expandbar'}).appendTo(elem) bar.append( - $('
', {class: 'expander'}).on('click', async (e) => { - await this.doExpand(obj.self.id, e) + $('
', {class: 'expander'}).on('click', async () => { + await this.doExpand(obj.self.id) }) ) @@ -289,37 +305,95 @@ class Treeview { } } - async onData(tree) { - const runID = JSON.stringify({name: 'onData', timestamp: Date.now()}) - console.time(runID) - try { - this.dom = new DOM(this.log, this.kc) - await this.onDataImpl(tree) - } finally { - console.timeEnd(runID) + async onPageID(ids) { + this.page_id = [].concat(ids) + + this.page_idx_id = null + const ns_id = ids.shift() + + for (const type of Object.values(IType)) { + const rvid = IndexID.composeReverseID(type, ids) + if (this.db.reverseID.has(rvid)) { + this.page_idx_id = this.db.reverseID.get(rvid) + } } - } - async onDataImpl(tree) { - this.log.debug('data', tree) - let root = $('
    ', {class: 'root'}).appendTo(this.root) + this.page_idx = null - const cats = this.kc.categories() + try { + if (!this.page_idx_id) { + throw new Error(`IndexID for path '${ids.join('/')}' not present in database`) + } + if (!IndexID.isClassy(this.page_idx_id.type)) { + this.log.info(`current page '${this.page_idx_id.join()}' is not classy. nothing to expand`) + + } else { + this.log.info(`classy page '${this.page_idx_id.join()}'`) - for (const top of tree) { - if (top.category.index === cats.get('index').index) { - continue + for (const ns of this.db.namespaces) { + if (ns.namespace[0] !== 'reference') continue + + this.log.debug(`checking Namespace '${ns.pretty_name()}'`, ns) + if (ns.indexes.has(this.page_idx_id)) { + this.page_idx = ns.indexes.get(this.page_idx_id) + } + } + + if (!this.page_idx) { + throw new Error(`Index for '${this.page_idx_id.join()}' not found in any of the 'reference' namespace`) + } + + const h = this.page_idx.in_header + this.log.info(`expanding current page header '${h.id.join()}'`, h, this.page_idx) + + // await this.dom.doExpand(h.id) + await this.dom.doExpand(h.id) + await this.dom.scrollAt(h.id) } - let e = $('
  • ', {class: 'top', 'data-top-id': top.namespace.namespace[0]}).appendTo(root) - e.append(await this.dom.makeTitle(top)) + } catch (e) { + this.log.error(`Failed to determine current page for id '${ids.join('/')}'. Sidebar will NOT work properly! (${e})`, ids) + } + } - if (top.category.index === cats.get('lang').index) { - this.processLangTop(top, e) - } else { - this.processTop(top, e) + async onData(db) { + this.db = db + this.tree = db.getTree(this.kc) + + // workaround name + for (let t of this.tree) { + if (t.root) { + const name = t.root.id.join() + const real_name = t.category.name + if (name !== real_name) { + this.log.warn(`got incorrect title '${name}'; expected = '${real_name}'. ignoring...`) + } } } + + this.dom = new DOM(this.log, this.kc) + await this.onDataImpl() + } + + async onDataImpl() { + this.log.debug('data', this.tree) + let root = $('
      ', {class: 'root'}).appendTo(this.root) + + const cats = this.kc.categories() + + root.append(await Promise.all( + this.tree.filter((top) => top.category.index !== cats.get('index').index).map(async (top) => { + let e = $('
    • ', {class: 'top', 'data-top-id': top.namespace.namespace[0]}) + e.append(await this.dom.makeTitle(top)) + + if (top.category.index === cats.get('lang').index) { + this.processLangTop(top, e) + } else { + this.processTop(top, e) + } + return e + }) + )) } async processTop(top, e) { diff --git a/js/kunai/wand.js b/js/kunai/wand.js index 8367b8c..60012e2 100644 --- a/js/kunai/wand.js +++ b/js/kunai/wand.js @@ -3,6 +3,8 @@ import {NetworkError} from './error' import {default as Numeral} from 'numeral' import {Logger} from 'nagato' +import URL from 'url-parse' + class APIError extends NetworkError { constructor() {