Angular中使用MathJax遇到的一些問題
前言
話說我本來是傾向于 KaTeX 的,因為我感覺他很快,而且 MathJax 似乎很難配。但是大家表示對缺少功能的 KaTeX 并無好感,給我提供了一些鉆研 MathJax 的動力。
MathJax簡介
MathJax是一款運行在瀏覽器中的開源數(shù)學符號渲染引擎,使用MathJax可以方便的在瀏覽器中顯示數(shù)學公式,不需要使用圖片。目前,MathJax可以解析Latex、MathML和ASCIIMathML的標記語言。MathJax項目于2009年開始,發(fā)起人有American Mathematical Society, Design Science等,還有眾多的支持者,個人感覺MathJax會成為今后數(shù)學符號渲染引擎中的主流,也許現(xiàn)在已經(jīng)是了。
個人其實也不算鉆研,因為實際上 MathJax 很簡單,調用 MathJax.Hub.Queue(['Typeset', MathJax.Hub, this.element.nativeElement]); 就可以渲染一個元素(這個 this.element.nativeElement 是從 Angular 中調用它 DOM 的語法),這個 .Queue 實際上是 MathJax 自己實現(xiàn)的回調格式,語法非常清奇,參數(shù)個數(shù)不定,每個都是數(shù)組,代表一個回調,順序執(zhí)行。比如這個 ['Typeset', MathJax.Hub, this.element.nativeElement] ,第一個元素是方法名,第二個元素是 this ,之后的元素都是參數(shù)……
我們可以看到這個就相當于執(zhí)行 MathJax.Hub.Typeset(this.element.nativeElement) ,那為啥不執(zhí)行這個?因為這方法是同步的,會導致頁面十分卡。于是 MathJax 就自己封裝了一個異步隊列(它的 API 可能幾百年沒改了)
我們說回 Angular。因為要用 markdown,我的思路是用 marked 封裝一個 directive。那么我們就應該在 marked 渲染完成之后用 MathJax 去 Typeset 這個組件。但真的這樣做了,卻產(chǎn)生了奇妙的效果——切換頁面之后,要等將近一分鐘才開始渲染。我在它的隊列里放了幾個 log,發(fā)現(xiàn)每個元素都被 queue 了 4 次,幾十個元素,難怪要一分鐘才開始渲染下一頁的內(nèi)容,即使大部分 markdown 里面根本沒有數(shù)學。
這時候我開始灰心了,這個問題就沒有解決辦法了嗎?絕望之時,我想到能不能直接 Typeset document,結果是可以的,而且十分快。所以渲染并不慢,可能是渲染的初始化過程比較慢。那么這時候方案就出來了,我們可以盡量減少渲染次數(shù),同時只渲染 document。只要這個渲染還在進行,那么有再多的元素 queue 上來,我們也只當作 queue 了一次。
于是我就寫了這么個 service:
@Injectable()
export class MathjaxService {
public isQueued = false;
public isRunning = false;
window: any;
constructor(@Inject(PLATFORM_ID) private platformId: Object) {
if (isPlatformBrowser(this.platformId)) {
this.window = window as any;
}
}
finishRunning() {
this.isRunning = false;
if (this.isQueued) {
this.queueChange();
}
}
queueChange() {
if (this.isRunning) {
this.isQueued = true;
} else {
this.isQueued = false;
this.isRunning = true;
if (isPlatformBrowser(this.platformId)) {
if (this.window.MathJax) {
this.window.MathJax.Hub.Config({
messageStyle: 'none',
tex2jax: {
// preview: 'none',
inlineMath: [['$', '$']],
processEscapes: true
}
});
this.window.MathJax.Hub.Queue(['log', console, 'start'], ['Typeset', this.window.MathJax.Hub, document], ['log', console, 'end'], ['finishRunning', this]);
}
} else {
this.finishRunning();
}
}
}
}
事實證明,它能圓滿完成任務,它也就是現(xiàn)在運行在 這個網(wǎng)站 上的代碼。
總結
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關文章
Angularjs Ng_repeat中實現(xiàn)復選框選中并顯示不同的樣式方法
今天小編就為大家分享一篇Angularjs Ng_repeat中實現(xiàn)復選框選中并顯示不同的樣式方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09
詳解angularjs中如何實現(xiàn)控制器和指令之間交互
本篇文章主要介紹了詳解angularjs中如何實現(xiàn)控制器和指令之間交互,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-05-05
angularJs中json數(shù)據(jù)轉換與本地存儲的實例
今天小編就為大家分享一篇angularJs中json數(shù)據(jù)轉換與本地存儲的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-10-10
AngularJS 使用ng-repeat報錯 [ngRepeat:dupes]
這篇文章主要介紹了AngularJS 使用ng-repeat報錯 [ngRepeat:dupes] 的相關資料,需要的朋友可以參考下2017-01-01

