From f2a4162c81715deaa118dd33c094e4dcabd08b1c Mon Sep 17 00:00:00 2001 From: Koichi Murase Date: Sun, 29 May 2022 21:50:47 +0900 Subject: [PATCH 1/6] =?UTF-8?q?kunai.js:=20crsearch.json=E3=81=AE=E4=BD=8D?= =?UTF-8?q?=E7=BD=AE=E3=82=92=E8=87=AA=E5=8B=95=E3=81=A7=E7=89=B9=E5=AE=9A?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/kunai.js | 50 +++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 49 insertions(+), 1 deletion(-) diff --git a/js/kunai.js b/js/kunai.js index 671c1ee..95e7d37 100644 --- a/js/kunai.js +++ b/js/kunai.js @@ -116,10 +116,58 @@ class Kunai { await this.initSidebar() + // Dynamically set the base_url + const dynamic_base_url = (() => { + // Determine the location of the website base + const current_script = document.currentScript || document.querySelector('script[src*="static/kunai/js/kunai.js"]') + if (current_script) { + // Try to determine the base_url based on the location of this script + // ({base_url}/static/kunai/js/kunai.js). + const url_kunai = current_script.getAttribute("src") + const url = url_kunai.replace(/\bstatic\/kunai\/js\/kunai\.js([?#].*)?$/, "") + if (url != url_kunai) return url == "" ? "/" : url + } + // Fallback case assuming that the website is hosted at the top level + return "/" + })() + + const database_url = (() => { + // Determine the location of the database file "crsearch.json". + const current_script = document.currentScript || document.querySelector('script[src*="kunai/js/kunai.js"]') + if (current_script) { + // Try to download crsearch.json from the project website for local + // HTML files. When a HTML in a local file system is directly opened + // in a Web browser, "static/crsearch/crsearch.json" cannot be read + // through XHR due to the CORS (cross-origin resource sharing) policy + // for the local files. We instead try to download "crsearch.json" + // from the project website, which is assumed to be stored in or in . + if (/^file:\/\//.test(current_script.src)) { + const meta = document.querySelector('meta[name="twitter:url"]') || document.querySelector('meta[property="og:url"]') + if (meta && meta.content) { + const m = meta.content.toString().match(/^https?:\/\/[^/]*\//) + if (m) return m[0] + "static/crsearch/crsearch.json" + } + } + + // Try to determine the position of crsearch.json + // ({base_url}/static/crsearch/crsearch.json) based on the location of + // this script ({base_url}/static/kunai/js/kunai.js). + const url_kunai = current_script.getAttribute("src") + const url = url_kunai.replace(/\bkunai\/js\/kunai\.js([?#].*)?$/, "crsearch/crsearch.json") + if (url != url_kunai) return url + } + + // Fallback case assuming that the website is hosted at the top level + return "/static/crsearch/crsearch.json" + })(); + let crs = new CRSearch({ onDatabase: this.onDatabase.bind(this), + base_url: dynamic_base_url }) - crs.database('/static/crsearch/crsearch.json') + crs.database(database_url) let e = $('.crsearch') await crs.searchbox(e) From 93678a38eaf3432d7b6d68b67097d3a60a1fe54b Mon Sep 17 00:00:00 2001 From: Koichi Murase Date: Fri, 3 Jun 2022 18:39:23 +0900 Subject: [PATCH 2/6] =?UTF-8?q?navbar.css:=20cpprefjp=E3=82=A2=E3=82=A4?= =?UTF-8?q?=E3=82=B3=E3=83=B3=E3=82=92data=E3=82=B9=E3=82=AD=E3=83=BC?= =?UTF-8?q?=E3=83=A0=E3=81=A7=E5=9F=8B=E3=82=81=E8=BE=BC=E3=82=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit できるだけ多くのブラウザで見られるように以下の記事 [1] を参考にした。 - [1] https://yukiyuriweb.com/2016/10/08/using-data-uri-scheme-for-svg-in-background-image-compatible-with-ie/ URL エンコードに関連する変換は以下を行う。 # 行頭・行末の空白を削除 /^$/d s/^[[:space:]]*// s/[[:space:]]*$/ / s/> $/>/ # 文字の置換 s/"/'/g s//%3E/g s/#/%23/g その後に tr -d '\n' で改行を削除した。 --- css/kunai/site/navbar.css | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/css/kunai/site/navbar.css b/css/kunai/site/navbar.css index b64d515..2e2f413 100644 --- a/css/kunai/site/navbar.css +++ b/css/kunai/site/navbar.css @@ -48,7 +48,7 @@ nav[role="navigation"] { height: 0; padding: 32px 32px 0 0; margin: 0 4px 0 0; - background-image: url(https://cpprefjp.github.io/static/original-icons/cpprefjp-icon-v2.0-transparent.png); + background-image: url("data:image/svg+xml;charset=utf8,%3Csvg width='26.000004mm' height='26.000004mm' viewBox='0 0 26.000004 26.000004' version='1.1' id='svg1' xmlns='http://www.w3.org/2000/svg' xmlns:svg='http://www.w3.org/2000/svg'%3E%3Cdefs id='defs1'%3E%3CclipPath clipPathUnits='userSpaceOnUse' id='clipPath12'%3E%3Crect style='fill:%232ca9e1;stroke:none;stroke-width:0.45729;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none;stroke-dashoffset:0' id='rect13' width='26' height='26' x='92.05806' y='113.9539' /%3E%3C/clipPath%3E%3C/defs%3E%3Cg id='layer1' transform='translate(-92.058062,-113.95391)'%3E%3Cg id='g12' clip-path='url(%23clipPath12)'%3E%3Cpath id='path2-6' style='fill:none;stroke:%2300a3af;stroke-width:1.05833;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none' d='m 92.803909,132.76018 -9.95492,5.62988 -9.85309,-5.80628 0.10185,-11.43616 9.95493,-5.62987 9.85308,5.80627 z' /%3E%3Cpath id='path2-2' style='fill:none;stroke:%2300a3af;stroke-width:1.05833;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none' d='m 137.01837,132.76018 -9.95492,5.62988 -9.85309,-5.80628 0.10185,-11.43616 9.95493,-5.62988 9.85308,5.80628 z' /%3E%3Cpath id='path2-9' style='fill:none;stroke:%2300a3af;stroke-width:2.11666;stroke-linecap:round;stroke-linejoin:round' d='m 112.76826,132.84545 -7.86126,4.70835 -9.038106,-5.38578 0.09342,-10.60799 9.131526,-5.22216 7.51379,4.6151' /%3E%3Cpath style='fill:%232ca9e1;stroke:none;stroke-width:0.945904;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none;stroke-dashoffset:0' id='path3-6' d='m 68.619675,152.01895 -1.306347,-4.85182 -4.851829,-1.30635 4.851829,-1.30635 1.306347,-4.85183 1.306346,4.85183 4.851829,1.30635 -4.851829,1.30635 z' transform='matrix(0.8119288,0,0,0.82328669,47.479406,6.8605971)' /%3E%3Cpath style='fill:%232ca9e1;stroke:none;stroke-width:0.945904;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none;stroke-dashoffset:0' id='path3-6-1' d='m 68.619675,152.01895 -1.306347,-4.85182 -4.851829,-1.30635 4.851829,-1.30635 1.306347,-4.85183 1.306346,4.85183 4.851829,1.30635 -4.851829,1.30635 z' transform='matrix(0.8119288,0,0,0.82328669,55.018249,6.8605971)' /%3E%3Cg id='g5' transform='translate(0.20305)'%3E%3Cpath id='path2-6-4' style='fill:none;stroke:%2300a3af;stroke-width:1.05833;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none' d='m 103.65447,152.06483 -9.954911,5.62988 -9.85309,-5.80628 0.10185,-11.43616 9.95493,-5.62987 9.853071,5.80627 z' /%3E%3Cpath id='path2-5' style='fill:none;stroke:%2300a3af;stroke-width:1.05833;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none' d='m 125.7617,152.06483 -9.95492,5.62988 -9.85308,-5.80628 0.10184,-11.43616 9.95493,-5.62988 9.85308,5.80628 z' /%3E%3C/g%3E%3Cg id='g6' transform='translate(0.0998685)'%3E%3Cpath id='path2-6-4-9' style='fill:none;stroke:%2300a3af;stroke-width:1.05833;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none' d='m 103.75765,113.45553 -9.954908,5.62988 -9.85309,-5.80628 0.10185,-11.43616 9.95493,-5.629874 9.853068,5.806274 z' /%3E%3Cpath id='path2-5-1' style='fill:none;stroke:%2300a3af;stroke-width:1.05833;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none' d='m 125.86488,113.45553 -9.95492,5.62988 -9.85308,-5.80628 0.10184,-11.43616 9.95493,-5.629884 9.85308,5.806284 z' /%3E%3C/g%3E%3C/g%3E%3C/g%3E%3C/svg%3E"); background-size: contain; background-repeat: no-repeat; } From 8e1dbf10f11eb36d04690aa4c3e229ee6c03fc3c Mon Sep 17 00:00:00 2001 From: Koichi Murase Date: Mon, 6 Jun 2022 01:53:43 +0900 Subject: [PATCH 3/6] =?UTF-8?q?kunai.js:=20file://=20=E3=81=A7=E3=81=AFJSO?= =?UTF-8?q?NP=E3=81=A7crsearch=E3=82=92=E8=AA=AD=E8=BE=BC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/kunai.js | 23 +++++++++++++++-------- 1 file changed, 15 insertions(+), 8 deletions(-) diff --git a/js/kunai.js b/js/kunai.js index 95e7d37..02f6fe6 100644 --- a/js/kunai.js +++ b/js/kunai.js @@ -135,15 +135,22 @@ class Kunai { // Determine the location of the database file "crsearch.json". const current_script = document.currentScript || document.querySelector('script[src*="kunai/js/kunai.js"]') if (current_script) { - // Try to download crsearch.json from the project website for local - // HTML files. When a HTML in a local file system is directly opened - // in a Web browser, "static/crsearch/crsearch.json" cannot be read - // through XHR due to the CORS (cross-origin resource sharing) policy - // for the local files. We instead try to download "crsearch.json" - // from the project website, which is assumed to be stored in or in . + // A special care is needed for local HTML files (file://...). When a + // HTML in a local file system is directly opened in a Web browser, + // "static/crsearch/crsearch.json" cannot be read using XHR due to the + // CORS (cross-origin resource sharing) policy for the local files. if (/^file:\/\//.test(current_script.src)) { + const url_kunai = current_script.getAttribute("src") + + // When the current script file (kunai.js) is located in an expected + // path in the tree, we try to load the local database file + // "crsearch/crsearch.js" in JSONP format. + const url = url_kunai.replace(/\bkunai\/js\/kunai\.js([?#].*)?$/, "crsearch/crsearch.js") + if (url != url_kunai) return url + + // Try to download "crsearch.json" from the project website, which is + // assumed to be stored in + // or in . const meta = document.querySelector('meta[name="twitter:url"]') || document.querySelector('meta[property="og:url"]') if (meta && meta.content) { const m = meta.content.toString().match(/^https?:\/\/[^/]*\//) From ef44d541bc016b98cbab5a110e04b1ba2de63c70 Mon Sep 17 00:00:00 2001 From: Koichi Murase Date: Mon, 6 Jun 2022 07:41:05 +0900 Subject: [PATCH 4/6] =?UTF-8?q?kunai/ui:=20=E8=84=B1=E5=AD=97=E4=BF=AE?= =?UTF-8?q?=E6=AD=A3=20"initial{=20=3D>=20i}zing"?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/kunai/ui/content.js | 2 +- js/kunai/ui/sidebar.js | 2 +- js/kunai/ui/treeview.js | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/js/kunai/ui/content.js b/js/kunai/ui/content.js index 68f3027..29f55e4 100644 --- a/js/kunai/ui/content.js +++ b/js/kunai/ui/content.js @@ -12,7 +12,7 @@ const _hitElementRects = (elem, x, y) => { class Content { constructor(log) { this.log = log.makeContext('Content') - this.log.debug('initialzing...') + this.log.debug('initializing...') this.log.debug(`found ${Badge.sanitize($('main[role="main"] div[itemtype="http://schema.org/Article"] .content-body span.cpp'))} badges`) diff --git a/js/kunai/ui/sidebar.js b/js/kunai/ui/sidebar.js index b00301d..5caaecc 100644 --- a/js/kunai/ui/sidebar.js +++ b/js/kunai/ui/sidebar.js @@ -5,7 +5,7 @@ import {KC} from 'crsearch' class Sidebar { constructor(log) { this.log = log.makeContext('Sidebar') - this.log.info('initialzing...') + this.log.info('initializing...') this.kc = new KC.Config({ 'article.md': require('../../../kunai_configs/current/article.md').default, diff --git a/js/kunai/ui/treeview.js b/js/kunai/ui/treeview.js index 30ad613..2c10993 100644 --- a/js/kunai/ui/treeview.js +++ b/js/kunai/ui/treeview.js @@ -283,7 +283,7 @@ class Treeview { this.opts = Object.assign({}, opts) this.legacy = this.opts.legacy - this.log.debug('initialzing...') + this.log.debug('initializing...') if (this.legacy) { const c = Badge.sanitize(this.e.find('.cpp-sidebar')) From b66c4d78dd622ea8a8088313651adc7f4aa9090c Mon Sep 17 00:00:00 2001 From: Koichi Murase Date: Mon, 6 Jun 2022 08:21:55 +0900 Subject: [PATCH 5/6] =?UTF-8?q?badge.js:=20C++=E3=83=90=E3=83=BC=E3=82=B8?= =?UTF-8?q?=E3=83=A7=E3=83=B3=E3=83=90=E3=83=83=E3=82=B8=E3=81=AEbase=5Fur?= =?UTF-8?q?l=E3=82=92=E9=81=85=E5=BB=B6=E3=81=A7=E5=86=8D=E8=A8=AD?= =?UTF-8?q?=E5=AE=9A?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/kunai.js | 1 + js/kunai/ui.js | 3 ++- js/kunai/ui/badge.js | 25 +++++++++++++++++++------ 3 files changed, 22 insertions(+), 7 deletions(-) diff --git a/js/kunai.js b/js/kunai.js index 02f6fe6..db01360 100644 --- a/js/kunai.js +++ b/js/kunai.js @@ -107,6 +107,7 @@ class Kunai { async onDatabase(db) { // this.log.debug(`onDatabase`, db) + UI.Badge.onDatabase(db) await this.ui.sidebar.onDatabase(db) await this.ui.sidebar.treeview.onPageID(this.meta.page_id) } diff --git a/js/kunai/ui.js b/js/kunai/ui.js index 98d0db2..1745f55 100644 --- a/js/kunai/ui.js +++ b/js/kunai/ui.js @@ -1,4 +1,5 @@ export {Content} from './ui/content' export {Sidebar} from './ui/sidebar' export {Navbar} from './ui/navbar' - +import * as Badge from './ui/badge' +export {Badge} diff --git a/js/kunai/ui/badge.js b/js/kunai/ui/badge.js index adc0f14..0abe2e0 100644 --- a/js/kunai/ui/badge.js +++ b/js/kunai/ui/badge.js @@ -1,3 +1,13 @@ +let base_url = null +const unresolved_links = [] + +const onDatabase = (db) => { + base_url = db.base_url.toString() + for (let a_elem of unresolved_links) + a_elem.attr('href', base_url.replace(/\/$/, '') + a_elem.attr('href')) + unresolved_links.length = 0 +} + const sanitize = (badges) => { let i = 0 @@ -46,15 +56,18 @@ const sanitize = (badges) => { const lang_path = cppv ? `/lang/cpp${cppv}` : named_version ? `/lang/${named_version}` : `/lang` + const a_elem = $('', {href: `${lang_path}.html`}) + .append($('')) + // .append($('').text(clean_txt)) + .appendTo(b.empty()) - b.empty().append( - $('', {href: `${lang_path}.html`}) - .append($('')) - // .append($('').text(clean_txt)) - ) + if (base_url) + a_elem.attr('href', base_url.replace(/\/$/, '') + a_elem.attr('href')) + else + unresolved_links.push(a_elem) } return i } -export {sanitize} +export {onDatabase, sanitize} From 9c16b0834006ee9d4db2d0c5adeda0699b11e2a3 Mon Sep 17 00:00:00 2001 From: Koichi Murase Date: Sat, 18 May 2024 00:39:13 +0900 Subject: [PATCH 6/6] =?UTF-8?q?kunai.js:=20=E3=82=AA=E3=83=B3=E3=83=A9?= =?UTF-8?q?=E3=82=A4=E3=83=B3URL=E3=82=92=E3=83=87=E3=83=BC=E3=82=BF?= =?UTF-8?q?=E3=83=99=E3=83=BC=E3=82=B9=E3=81=AB=E6=8C=87=E5=AE=9A=20(Googl?= =?UTF-8?q?e=E6=A4=9C=E7=B4=A2=E5=AF=BE=E8=B1=A1=E3=82=B5=E3=82=A4?= =?UTF-8?q?=E3=83=88=E3=81=AB=E4=BD=BF=E7=94=A8)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/kunai.js | 26 +++++++++++++++++--------- 1 file changed, 17 insertions(+), 9 deletions(-) diff --git a/js/kunai.js b/js/kunai.js index db01360..852a2e3 100644 --- a/js/kunai.js +++ b/js/kunai.js @@ -132,6 +132,18 @@ class Kunai { return "/" })() + // Determine the project website URL, which is assumed to be stored in + // or in . + const online_base_url = (() => { + const meta = document.querySelector('meta[name="twitter:url"]') || document.querySelector('meta[property="og:url"]') + if (meta && meta.content) { + const m = meta.content.toString().match(/^https?:\/\/[^/]*\//) + if (m) return m[0] + } + return null + })() + const database_url = (() => { // Determine the location of the database file "crsearch.json". const current_script = document.currentScript || document.querySelector('script[src*="kunai/js/kunai.js"]') @@ -149,14 +161,9 @@ class Kunai { const url = url_kunai.replace(/\bkunai\/js\/kunai\.js([?#].*)?$/, "crsearch/crsearch.js") if (url != url_kunai) return url - // Try to download "crsearch.json" from the project website, which is - // assumed to be stored in - // or in . - const meta = document.querySelector('meta[name="twitter:url"]') || document.querySelector('meta[property="og:url"]') - if (meta && meta.content) { - const m = meta.content.toString().match(/^https?:\/\/[^/]*\//) - if (m) return m[0] + "static/crsearch/crsearch.json" - } + // Try to download "crsearch.json" from the project website. + if (online_base_url) + return online_base_url + "static/crsearch/crsearch.json" } // Try to determine the position of crsearch.json @@ -173,7 +180,8 @@ class Kunai { let crs = new CRSearch({ onDatabase: this.onDatabase.bind(this), - base_url: dynamic_base_url + base_url: dynamic_base_url, + online_base_url: online_base_url }) crs.database(database_url)