LoginSignup
4
4

More than 1 year has passed since last update.

AnkiのJavaScriptライブラリの読み込み

Last updated at Posted at 2022-12-21

はじめに

これは、JavaScript Advent Calendar 2022の21日目の記事となります。

大学入学共通テストまで2ヶ月ほどになりました。たくさんの情報を効率よく覚えたい、タイパよく暗記したくありませんか?
今回紹介するのは暗記アプリ「Anki」のJavaScriptを使用したカスタマイズとなります。

以前、JavaScriptを使用したカスタマイズとして英作文の組み立てを作成しました。
これをもっとリッチな表現にしたいと思っています。

Ankiとは

Ankiとは、Damien Elmes氏がプログラミング言語Pythonを用いて開発した、暗記を目的としたオープンソフトウェアです。その名の通り、日本語の「暗記」から名付けられており、フラッシュカードを自由にカスタマイズできる機能が豊富なことで知られています。

「忘却曲線」という単語を聞いたことはありませんでしょうか?、時間が経過するにつれて記憶は薄れていくという理論です。記憶は1日経過しただけで半分以上忘れてしまいます。
しかし、適切なタイミングで復習すれば記憶の忘却のスピードは緩やかになり、結果としてすぐに忘れてしまう短期記憶から、時間が経っても忘れない、長期記憶に移行するのです。

Ankiは、適切なタイミングで復習出来るように出題をコントロールしてくれるアプリです。

JavaScriptによる機能拡張

AnkiはPythonを利用したアドオンを作成することが出来るのですが、PC向け(Windows、Mac、Linux)のみで、モバイル向けは対象外です。

JavaScriptによる機能拡張は、PCやモバイルで使用可能です。
AnkiのPC版はQWebEngineを使用しています。ES6 modulesに対応しています。

CDNの利用

オンライン環境(インターネット接続が可能)であれば、JavaScriptライブラリーをCDNから取得し利用すると便利です。

collection.mediaフォルダの利用

オフライン環境の場合、JavaScriptライブラリーをcollection.mediaフォルダに保存しておくとでもオフライン状態でも動作するようになります。

collection.mediaフォルダは、すべてのクライアント (Anki Desktop、AnkiMobile など) 間で同期される仕組みとなっています。
この時だけインターネット接続する必要があります。

使用方法

Ankiのネームルールとしてファイル名の先頭にアンダースコアを追加して、collection.mediaフォルダに保存します。

await injectScript('https://cdnjs.cloudflare.com/ajax/libs/marked/0.3.5/marked.js');
↓
await injectScript('_marked.js');

動的ロード

カード テンプレートのインライン Javascript は Anki 2.1 で非同期に評価されるため、外部ライブラリを使用する関数は意図したとおりに動作しない場合があります。
これに対するいくつかの回避策は、インライン スクリプトを使用せず、js ファイルを動的にロードすることです。
ES6 Promise APIを使用して動的スクリプト ローダーを介して外部ライブラリをロードする方法になります。

例として Markdownで書かれた内容をHTMLに変換するライブラリー「marked.js」とコードをシンタックスハイライトする(highlight.js)を利用して
FibonacciのPythonコードを表示させてみます。

image.png

{{FrontSide}}

<script>
var injectScript = (src) => {
    return new Promise((resolve, reject) => {
        const script = document.createElement('script');
        script.src = src;
        script.async = true;
        script.onload = resolve;
        script.onerror = reject;
        document.head.appendChild(script);
    });
};

var injectCSS = (src) => {
    return new Promise((resolve, reject) => {
        const link = document.createElement('link');
        link.rel = "stylesheet";
        link.href = src
        link.async = true;
        link.onload = resolve;
        link.onerror = reject;
        document.head.appendChild(link);
    });
};

(async () => {
    if (!('marked' in globalThis)) {
        await injectCSS('https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.6.0/build/styles/default.min.css');
        await injectScript('https://cdnjs.cloudflare.com/ajax/libs/marked/0.3.5/marked.js');
        await injectScript('https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.6.0/build/highlight.min.js');
    }

    document.querySelectorAll('.md').forEach((el) => {
        var highlight = function(code, lang, callback){
            return hljs.highlight('py',code).value;
        }

        var renderer = new marked.Renderer;
        var html = marked(el.innerText, {
            highlight: highlight,
            renderer: renderer,
        });
        el.innerHTML = html;
    });
})();
</script>

<hr id=answer>

<pre class="md" style="text-align: left;">
{{裏面}}
</pre>

質問と回答の共有

window オブジェクトが Anki カードの質問と回答の呼び出し間で共有されると述べています。
質問で乱数を生成し、同じ値が回答で使用できることを意味します。

【2023/01/03追記】

最後に

今回、英語の整序問題(並び替え)をリッチな表現にしたいと思って色々調べました。
何かのお役に立てれば幸いです。

4
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
4