소스 검색

Добавлена загрузка книги на клиента, сопутствующие компоненты, классы и модули

Book Pauk 6 년 전
부모
커밋
dd640f2d40

+ 43 - 0
client/api/reader.js

@@ -0,0 +1,43 @@
+import axios from 'axios';
+import {sleep} from '../share/utils';
+
+const api = axios.create({
+  baseURL: '/api/reader'
+});
+
+const workerApi = axios.create({
+  baseURL: '/api/worker'
+});
+
+class Reader {
+    async loadBook(url, callback) {
+        let response = await api.post('/load-book', {type: 'url', url});
+
+        const workerId = response.data.workerId;
+        if (!workerId)
+            throw new Error('Неверный ответ api');
+
+        let i = 0;
+        while (1) {// eslint-disable-line no-constant-condition
+            if (callback)
+                callback(response.data);
+            if (response.data.state == 'finish') {
+                let book = await axios.get(response.data.path, {});
+                return Object.assign({}, response.data, {data: book.data});
+            }
+            if (response.data.state == 'error') {
+                throw new Error(response.data.error);
+            }
+            if (i > 0)
+                await sleep(500);
+
+            i++;
+            if (i > 60) {
+                throw new Error('Слишком долгое время ожидания');
+            }
+            response = await workerApi.post('/get-state', {workerId});
+        }
+    }
+}
+
+export default new Reader();

+ 17 - 5
client/components/Reader/LoaderPage/LoaderPage.vue

@@ -11,8 +11,9 @@
             </el-input>
             <div class="space"></div>
             <el-button size="mini" @click="loadFle">
-                Загрузить файл
+                Загрузить файл с диска
             </el-button>
+            <span class="bottom-span"><pre>{{ loadState }}</pre></span>
         </div>
         <div class="part bottom">
             <span v-if="config.mode == 'omnireader'" class="bottom-span clickable" @click="openComments">Комментарии</span>
@@ -27,10 +28,13 @@
 import Vue from 'vue';
 import Component from 'vue-class-component';
 
+import readerApi from '../../../api/reader';
+
 export default @Component({
 })
 class LoaderPage extends Vue {
     bookUrl = null;
+    loadState = null;
 
     created() {
         this.commit = this.$store.commit;
@@ -53,10 +57,18 @@ class LoaderPage extends Vue {
         return `v${this.config.version}`;
     }
 
-    submitUrl() {
-        if (this.bookUrl)
-            //loadUrl()
-        ;
+    async submitUrl() {
+        if (this.bookUrl) {
+            try {
+                const book = await readerApi.loadBook(this.bookUrl, (state) => {
+                    this.loadState = state;
+                });
+
+                this.loadState = book;
+            } catch (e) {
+                this.loadState = e.message;
+            }
+        }
     }
 
     loadFle() {

+ 9 - 1
client/components/Reader/Reader.vue

@@ -54,10 +54,12 @@
 import Vue from 'vue';
 import Component from 'vue-class-component';
 import LoaderPage from './LoaderPage/LoaderPage.vue';
+import TextPage from './TextPage/TextPage.vue';
 
 export default @Component({
     components: {
-        LoaderPage
+        LoaderPage,
+        TextPage
     },
 })
 class Reader extends Vue {
@@ -77,6 +79,10 @@ class Reader extends Vue {
         return this.reader.fullScreenActive;
     }
 
+    get lastOpenedBook() {
+        return this.$store.getters['reader/lastOpenedBook'];
+    }
+
     buttonClick(button) {
         switch (button) {
             case 'loader': this.commit('reader/setLoaderActive', !this.loaderActive); break;
@@ -96,6 +102,8 @@ class Reader extends Vue {
     get pageActive() {
         let result = '';
 
+        if (this.lastOpenedBook)
+            result = 'TextPage';
         if (this.loaderActive)
             result = 'LoaderPage';
 

+ 34 - 0
client/components/Reader/TextPage/TextPage.vue

@@ -0,0 +1,34 @@
+<template>
+    <div class="main">
+        Text page 
+    </div>
+</template>
+
+<script>
+//-----------------------------------------------------------------------------
+import Vue from 'vue';
+import Component from 'vue-class-component';
+
+export default @Component({
+})
+class TextPage extends Vue {
+    bookUrl = null;
+
+    created() {
+        this.commit = this.$store.commit;
+        this.dispatch = this.$store.dispatch;
+        this.config = this.$store.state.config;
+    }
+
+    keyHook(event) {
+    }
+}
+//-----------------------------------------------------------------------------
+</script>
+<style scoped>
+.main {
+    flex: 1;
+    display: flex;
+    flex-direction: column;
+}
+</style>

+ 2 - 0
client/components/Reader/share/BookManager.js

@@ -0,0 +1,2 @@
+export default class BookManager {
+}

+ 0 - 0
client/components/Reader/share/BookParser.js


+ 723 - 0
client/components/Reader/share/easysax.js

@@ -0,0 +1,723 @@
+'use strict';
+
+/*
+new function() {
+    var parser = new EasySAXParser();
+
+    parser.ns('rss', { // or false
+        'http://search.yahoo.com/mrss/': 'media',
+        'http://www.w3.org/1999/xhtml': 'xhtml',
+        'http://www.w3.org/2005/Atom': 'atom',
+        'http://purl.org/rss/1.0/': 'rss',
+    });
+
+    parser.on('error', function(msgError) {
+    });
+
+    parser.on('startNode', function(elemName, getAttr, isTagEnd, getStrNode) {
+        var attr = getAttr();
+    });
+
+    parser.on('endNode', function(elemName, isTagStart, getStrNode) {
+    });
+
+    parser.on('textNode', function(text) {
+    });
+
+    parser.on('cdata', function(data) {
+    });
+
+
+    parser.on('comment', function(text) {
+        //console.log('--'+text+'--')
+    });
+
+    //parser.on('unknownNS', function(key) {console.log('unknownNS: ' + key)});
+    //parser.on('question', function() {}); // <? ... ?>
+    //parser.on('attention', function() {}); // <!XXXXX zzzz="eeee">
+
+    console.time('easysax');
+    for(var z=1000;z--;) {
+        parser.parse(xml)
+    };
+    console.timeEnd('easysax');
+};
+
+*/
+
+// << ------------------------------------------------------------------------ >> //
+
+EasySAXParser.entityDecode = xmlEntityDecode;
+export default EasySAXParser;
+
+var stringFromCharCode = String.fromCharCode;
+var objectCreate = Object.create;
+function NULL_FUNC() {};
+
+function entity2char(x) {
+    if (x === 'amp') {
+        return '&';
+    };
+
+    switch(x.toLocaleLowerCase()) {
+        case 'quot': return '"';
+        case 'amp': return '&'
+        case 'lt': return '<'
+        case 'gt': return '>'
+
+        case 'plusmn': return '\u00B1';
+        case 'laquo': return '\u00AB';
+        case 'raquo': return '\u00BB';
+        case 'micro': return '\u00B5';
+        case 'nbsp': return '\u00A0';
+        case 'copy': return '\u00A9';
+        case 'sup2': return '\u00B2';
+        case 'sup3': return '\u00B3';
+        case 'para': return '\u00B6';
+        case 'reg': return '\u00AE';
+        case 'deg': return '\u00B0';
+        case 'apos': return '\'';
+    };
+
+    return '&' + x + ';';
+};
+
+function replaceEntities(s, d, x, z) {
+    if (z) {
+        return entity2char(z);
+    };
+
+    if (d) {
+        return stringFromCharCode(d);
+    };
+
+    return stringFromCharCode(parseInt(x, 16));
+};
+
+function xmlEntityDecode(s) {
+    var s = ('' + s);
+
+    if (s.length > 3 && s.indexOf('&') !== -1) {
+        if (s.indexOf('&lt;') !== -1) {s = s.replace(/&lt;/g, '<');}
+        if (s.indexOf('&gt;') !== -1) {s = s.replace(/&gt;/g, '>');}
+        if (s.indexOf('&quot;') !== -1) {s = s.replace(/&quot;/g, '"');}
+
+        if (s.indexOf('&') !== -1) {
+            s = s.replace(/&#(\d+);|&#x([0123456789abcdef]+);|&(\w+);/ig, replaceEntities);
+        };
+    };
+
+    return s;
+};
+
+function cloneMatrixNS(nsmatrix) {
+    var nn = objectCreate(null);
+    for (var n in nsmatrix) {
+        nn[n] = nsmatrix[n];
+    };
+    return nn;
+};
+
+
+function EasySAXParser(config) {
+    if (!this) {
+        return null;
+    };
+
+    var onTextNode = NULL_FUNC, onStartNode = NULL_FUNC, onEndNode = NULL_FUNC, onCDATA = NULL_FUNC, onError = NULL_FUNC, onComment, onQuestion, onAttention, onUnknownNS;
+    var is_onComment = false, is_onQuestion = false, is_onAttention = false, is_onUnknownNS = false;
+
+    var isAutoEntity = true; // делать "EntityDecode" всегда
+    var entityDecode = xmlEntityDecode;
+    var hasSurmiseNS = false;
+    var isNamespace = false;
+    var returnError = null;
+    var parseStop = false; // прервать парсер
+    var defaultNS;
+    var nsmatrix = null;
+    var useNS;
+    var xml = ''; // string
+
+
+    this.setup = function (op) {
+        for (var name in op) {
+            switch(name) {
+                case 'entityDecode': entityDecode = op.entityDecode || entityDecode; break;
+                case 'autoEntity': isAutoEntity = !!op.autoEntity; break;
+                case 'defaultNS': defaultNS = op.defaultNS || null; break;
+                case 'ns': isNamespace = !!(useNS = op.ns || null); break;
+                case 'on':
+                    var listeners = op.on;
+                    for (var ev in listeners) {
+                        this.on(ev, listeners[ev]);
+                    };
+                break;
+            };
+        };
+    };
+
+    this.on = function(name, cb) {
+        if (typeof cb !== 'function') {
+            if (cb !== null) {
+                throw error('required args on(string, function||null)');
+            };
+        };
+
+        switch(name) {
+            case 'startNode': onStartNode = cb || NULL_FUNC; break;
+            case 'textNode': onTextNode = cb || NULL_FUNC; break;
+            case 'endNode': onEndNode = cb || NULL_FUNC; break;
+            case 'error': onError = cb || NULL_FUNC; break;
+            case 'cdata': onCDATA = cb || NULL_FUNC; break;
+
+            case 'unknownNS': onUnknownNS = cb; is_onUnknownNS = !!cb; break;
+            case 'attention': onAttention = cb; is_onAttention = !!cb; break; // <!XXXXX zzzz="eeee">
+            case 'question': onQuestion = cb; is_onQuestion = !!cb; break; // <? ....  ?>
+            case 'comment': onComment = cb; is_onComment = !!cb; break;
+        };
+    };
+
+    this.ns = function(root, ns) {
+        if (!root) {
+            isNamespace = false;
+            defaultNS = null;
+            useNS = null;
+            return this;
+        };
+
+        if (!ns || typeof root !== 'string') {
+            throw error('required args ns(string, object)');
+        };
+
+        isNamespace = !!(useNS = ns || null);
+        defaultNS = root || null;
+
+        return this;
+    };
+
+    this.parse = function(_xml) {
+        if (typeof _xml !== 'string') {
+            return 'required args parser(string)'; // error
+        };
+
+        returnError = null;
+        xml = _xml;
+
+        if (isNamespace) {
+            nsmatrix = objectCreate(null);
+            nsmatrix.xmlns = defaultNS;
+
+            parse();
+
+            nsmatrix = null;
+
+        } else {
+            parse();
+        };
+
+        parseStop = false;
+        attrRes = true;
+        xml = '';
+
+        return returnError;
+    };
+
+    this.stop = function() {
+        parseStop = true;
+    };
+
+    if (config) {
+        this.setup(config);
+    };
+
+    // -----------------------------------------------------
+
+
+    var stringNodePosStart; // number
+    var stringNodePosEnd; // number
+    var attrStartPos; // number начало позиции атрибутов в строке attrString <(div^ class="xxxx" title="sssss")/>
+    var attrString; // строка атрибутов <(div class="xxxx" title="sssss")/>
+    var attrRes; // закешированный результат разбора атрибутов , null - разбор не проводился, object - хеш атрибутов, true - нет атрибутов, false - невалидный xml
+
+    /*
+        парсит атрибуты по требованию. Важно! - функция не генерирует исключения.
+
+        если была ошибка разбора возврашается false
+        если атрибутов нет и разбор удачен то возврашается true
+        если есть атрибуты то возврашается обьект(хеш)
+    */
+
+    function getAttrs() {
+        if (attrRes !== null) {
+            return attrRes;
+        };
+
+        var xmlnsAlias;
+        var nsAttrName;
+        var attrList = isNamespace && hasSurmiseNS ? [] : null;
+        var i = attrStartPos + 1; // так как первый символ уже был проверен
+        var s = attrString;
+        var l = s.length;
+        var hasNewMatrix;
+        var newalias;
+        var value;
+        var alias;
+        var name;
+        var res = {};
+        var ok;
+        var w;
+        var j;
+
+
+        for(; i < l; i++) {
+            w = s.charCodeAt(i);
+
+            if (w === 32 || (w < 14 && w > 8) ) { // \f\n\r\t\v
+                continue
+            };
+
+            if (w < 65 || w > 122 || (w > 90 && w < 97) ) { // недопустимые первые символы
+                if (w !== 95 && w !== 58) { // char 95"_" 58":"
+                    return attrRes = false; // error. invalid first char
+                };
+            };
+
+            for(j = i + 1; j < l; j++) { // проверяем все символы имени атрибута
+                w = s.charCodeAt(j);
+
+                if ( w > 96 && w < 123 || w > 64 && w < 91 || w > 47 && w < 59 || w === 45 || w === 95) {
+                    continue;
+                };
+
+                if (w !== 61) { // "=" == 61
+                    return attrRes = false; // error. invalid char "="
+                };
+
+                break;
+            };
+
+            name = s.substring(i, j);
+            ok = true;
+
+            if (name === 'xmlns:xmlns') {
+                return attrRes = false; // error. invalid name
+            };
+
+            w = s.charCodeAt(j + 1);
+
+            if (w === 34) {  // '"'
+                j = s.indexOf('"', i = j + 2 );
+
+            } else {
+                if (w !== 39) { // "'"
+                    return attrRes = false; // error. invalid char
+                };
+
+                j = s.indexOf('\'', i = j + 2 );
+            };
+
+            if (j === -1) {
+                return attrRes = false; // error. invalid char
+            };
+
+            if (j + 1 < l) {
+                w = s.charCodeAt(j + 1);
+
+                if (w > 32 || w < 9 || (w < 32 && w > 13)) {
+                    // error. invalid char
+                    return attrRes = false;
+                };
+            };
+
+
+            value = s.substring(i, j);
+            i = j + 1; // след. семвол уже проверен потому проверять нужно следуюший
+
+            if (isAutoEntity) {
+                value = entityDecode(value);
+            };
+
+            if (!isNamespace) { //
+                res[name] = value;
+                continue;
+            };
+
+            if (hasSurmiseNS) {
+                // есть подозрение что в атрибутах присутствует xmlns
+                newalias = (name !== 'xmlns'
+                    ? name.charCodeAt(0) === 120 && name.substr(0, 6) === 'xmlns:' ? name.substr(6) : null
+                    : 'xmlns'
+                );
+
+                if (newalias !== null) {
+                    alias = useNS[entityDecode(value)];
+                    if (is_onUnknownNS && !alias) {
+                        alias = onUnknownNS(value);
+                    };
+
+                    if (alias) {
+                        if (nsmatrix[newalias] !== alias) {
+                            if (!hasNewMatrix) {
+                                nsmatrix = cloneMatrixNS(nsmatrix);
+                                hasNewMatrix = true;
+                            };
+
+                            nsmatrix[newalias] = alias;
+                        };
+                    } else {
+                        if (nsmatrix[newalias]) {
+                            if (!hasNewMatrix) {
+                                nsmatrix = cloneMatrixNS(nsmatrix);
+                                hasNewMatrix = true;
+                            };
+
+                            nsmatrix[newalias] = false;
+                        };
+                    };
+
+                    res[name] = value;
+                    continue;
+                };
+
+                attrList.push(name, value);
+                continue;
+            };
+
+            w = name.indexOf(':');
+            if (w === -1) {
+                res[name] = value;
+                continue;
+            };
+
+            if (nsAttrName = nsmatrix[name.substring(0, w)]) {
+                nsAttrName = nsmatrix['xmlns'] === nsAttrName ? name.substr(w + 1) : nsAttrName + name.substr(w);
+                res[nsAttrName + name.substr(w)] = value;
+            };
+        };
+
+
+        if (!ok) {
+            return attrRes = true;  // атрибутов нет, ошибок тоже нет
+        };
+
+        if (hasSurmiseNS)  {
+            xmlnsAlias = nsmatrix['xmlns'];
+
+            for (i = 0, l = attrList.length; i < l; i++) {
+                name = attrList[i++];
+
+                w = name.indexOf(':');
+                if (w !== -1) {
+                    if (nsAttrName = nsmatrix[name.substring(0, w)]) {
+                        nsAttrName = xmlnsAlias === nsAttrName ? name.substr(w + 1) : nsAttrName + name.substr(w);
+                        res[nsAttrName] = attrList[i];
+                    };
+                    continue;
+                };
+                res[name] = attrList[i];
+            };
+        };
+
+        return attrRes = res;
+    };
+
+    function getStringNode() {
+        return xml.substring(stringNodePosStart, stringNodePosEnd + 1);
+    };
+
+
+    function parse() {
+        var stacknsmatrix = [];
+        var nodestack = [];
+        var stopIndex = 0;
+        var _nsmatrix;
+        var isTagStart = false;
+        var isTagEnd = false;
+        var x, y, q, w;
+        var j = 0;
+        var i = 0;
+        var xmlns;
+        var elem;
+        var stop; // используется при разборе "namespace" . если встретился неизвестное пространство то события не генерируются
+
+
+        while(j !== -1) {
+            stop = stopIndex > 0;
+
+            if (xml.charCodeAt(j) === 60) { // "<"
+                i = j;
+            } else {
+                i = xml.indexOf('<', j);
+            };
+
+            if (i === -1) { // конец разбора
+                if (nodestack.length) {
+                    onError(returnError = 'unexpected end parse');
+                    return;
+                };
+
+                if (j === 0) {
+                    onError(returnError = 'missing first tag');
+                    return;
+                };
+
+                return;
+            };
+
+            if (j !== i && !stop) {
+                onTextNode(isAutoEntity ? entityDecode(xml.substring(j, i)) : xml.substring(j, i));
+                if (parseStop) {
+                    return;
+                };
+            };
+
+            w = xml.charCodeAt(i+1);
+
+            if (w === 33) { // "!"
+                w = xml.charCodeAt(i+2);
+                if (w === 91 && xml.substr(i + 3, 6) === 'CDATA[') { // 91 == "["
+                    j = xml.indexOf(']]>', i);
+                    if (j === -1) {
+                        onError(returnError = 'cdata');
+                        return;
+                    };
+
+                    if (!stop) {
+                        onCDATA(xml.substring(i + 9, j));
+                        if (parseStop) {
+                            return;
+                        };
+                    };
+
+                    j += 3;
+                    continue;
+                };
+
+
+                if (w === 45 && xml.charCodeAt(i + 3) === 45) { // 45 == "-"
+                    j = xml.indexOf('-->', i);
+                    if (j === -1) {
+                        onError(returnError = 'expected -->');
+                        return;
+                    };
+
+
+                    if (is_onComment && !stop) {
+                        onComment(isAutoEntity ? entityDecode(xml.substring(i + 4, j)) : xml.substring(i + 4, j));
+                        if (parseStop) {
+                            return;
+                        };
+                    };
+
+                    j += 3;
+                    continue;
+                };
+
+                j = xml.indexOf('>', i + 1);
+                if (j === -1) {
+                    onError(returnError = 'expected ">"');
+                    return;
+                };
+
+                if (is_onAttention && !stop) {
+                    onAttention(xml.substring(i, j + 1));
+                    if (parseStop) {
+                        return;
+                    };
+                };
+
+                j += 1;
+                continue;
+            };
+
+            if (w === 63) { // "?"
+                j = xml.indexOf('?>', i);
+                if (j === -1) { // error
+                    onError(returnError = '...?>');
+                    return;
+                };
+
+                if (is_onQuestion) {
+                    onQuestion(xml.substring(i, j + 2));
+                    if (parseStop) {
+                        return;
+                    };
+                };
+
+                j += 2;
+                continue;
+            };
+
+            j = xml.indexOf('>', i + 1);
+
+            if (j == -1) { // error
+                onError(returnError = 'unclosed tag'); // ...>
+                return;
+            };
+
+            attrRes = true; // атрибутов нет
+
+            //if (xml.charCodeAt(i+1) === 47) { // </...
+            if (w === 47) { // </...
+                isTagStart = false;
+                isTagEnd = true;
+
+                // проверяем что должен быть закрыт тотже тег что и открывался
+                if (!nodestack.length) {
+                    onError(returnError = 'close tag, requires open tag');
+                    return;
+                };
+
+                x = elem = nodestack.pop();
+                q = i + 2 + elem.length;
+
+                if (elem !== xml.substring(i + 2, q)) {
+                    onError(returnError = 'close tag, not equal to the open tag');
+                    return;
+                };
+
+                // проверим что в закрываюшем теге нет лишнего
+                for(; q < j; q++) {
+                    w = xml.charCodeAt(q);
+
+                    if (w === 32 || (w > 8 && w < 14)) {  // \f\n\r\t\v пробел
+                        continue;
+                    };
+
+                    onError(returnError = 'close tag');
+                    return;
+                };
+
+            } else {
+                if (xml.charCodeAt(j - 1) ===  47) { // .../>
+                    x = elem = xml.substring(i + 1, j - 1);
+
+                    isTagStart = true;
+                    isTagEnd = true;
+
+                } else {
+                    x = elem = xml.substring(i + 1, j);
+
+                    isTagStart = true;
+                    isTagEnd = false;
+                };
+
+                if (!(w > 96  && w < 123 || w > 64 && w < 91 || w === 95 || w === 58)) { // char 95"_" 58":"
+                    onError(returnError = 'first char nodeName');
+                    return;
+                };
+
+                for (q = 1, y = x.length; q < y; q++) {
+                    w = x.charCodeAt(q);
+
+                    if (w > 96 && w < 123 || w > 64 && w < 91 || w > 47 && w < 59 || w === 45 || w === 95) {
+                        continue;
+                    };
+
+                    if (w === 32 || (w < 14 && w > 8)) { // \f\n\r\t\v пробел
+                        attrRes = null; // возможно есть атирибуты
+                        elem = x.substring(0, q)
+                        break;
+                    };
+
+                    onError(returnError = 'invalid nodeName');
+                    return;
+                };
+
+                if (!isTagEnd) {
+                    nodestack.push(elem);
+                };
+            };
+
+
+            if (isNamespace) {
+                if (stop) { // потомки неизвестного пространства имен
+                    if (isTagEnd) {
+                        if (!isTagStart) {
+                            if (--stopIndex === 0) {
+                                nsmatrix = stacknsmatrix.pop();
+                            };
+                        };
+
+                    } else {
+                        stopIndex += 1;
+                    };
+
+                    j += 1;
+                    continue;
+                };
+
+                // добавляем в stacknsmatrix только если !isTagEnd, иначе сохраняем контекст пространств в переменной
+                _nsmatrix = nsmatrix;
+                if (!isTagEnd) {
+                    stacknsmatrix.push(nsmatrix);
+                };
+
+                if (isTagStart && (attrRes === null)) {
+                    if (hasSurmiseNS = x.indexOf('xmlns', q) !== -1) { // есть подозрение на xmlns
+                        attrStartPos = q;
+                        attrString = x;
+
+                        getAttrs();
+
+                        hasSurmiseNS = false;
+                    };
+                };
+
+                w = elem.indexOf(':');
+                if (w !== -1) {
+                    xmlns = nsmatrix[elem.substring(0, w)];
+                    elem = elem.substr(w + 1);
+
+                } else {
+                    xmlns = nsmatrix.xmlns;
+                };
+
+
+                if (!xmlns) {
+                    // элемент неизвестного пространства имен
+                    if (isTagEnd) {
+                        nsmatrix = _nsmatrix; // так как тут всегда isTagStart
+                    } else {
+                        stopIndex = 1; // первый элемент для которого не определено пространство имен
+                    };
+
+                    j += 1;
+                    continue;
+                };
+
+                elem = xmlns + ':' + elem;
+            };
+
+            stringNodePosStart = i;
+            stringNodePosEnd = j;
+
+            if (isTagStart) {
+                attrStartPos = q;
+                attrString = x;
+
+                onStartNode(elem, getAttrs, isTagEnd, getStringNode);
+                if (parseStop) {
+                    return;
+                };
+            };
+
+            if (isTagEnd) {
+                onEndNode(elem, isTagStart, getStringNode);
+                if (parseStop) {
+                    return;
+                };
+
+                if (isNamespace) {
+                    if (isTagStart) {
+                        nsmatrix = _nsmatrix;
+                    } else {
+                        nsmatrix = stacknsmatrix.pop();
+                    };
+                };
+            };
+
+            j += 1;
+        };
+    };
+};

+ 3 - 0
client/share/utils.js

@@ -0,0 +1,3 @@
+export function sleep(ms) {
+    return new Promise(resolve => setTimeout(resolve, ms));
+}

+ 23 - 1
client/store/modules/reader.js

@@ -1,11 +1,27 @@
+import Vue from 'vue';
+
 // initial state
 const state = {
     loaderActive: false,
     fullScreenActive: false,
+    openedBook: {},
 };
 
 // getters
-const getters = {};
+const getters = {
+    lastOpenedBook: (state) => {
+        let max = 0;
+        let result = null;
+        for (let bookKey in state.openedBook) {
+            const book = state.openedBook[bookKey];
+            if (book.touchTime > max) {
+                max = book.touchTime;
+                result = book;
+            }
+        }
+        return result;
+    },
+};
 
 // actions
 const actions = {};
@@ -18,6 +34,12 @@ const mutations = {
     setFullScreenActive(state, value) {
         state.fullScreenActive = value;
     },
+    addOpenedBook(state, value) {
+        Vue.set(state.openedBook, value.key, Object.assign({}, value, {touchTime: Date.now()}));
+    },
+    delOpenedBook(state, value) {
+        Vue.delete(state.openedBook, value.key);
+    }
 };
 
 export default {