diff --git a/client/api/reader.js b/client/api/reader.js
new file mode 100644
index 00000000..b1f44405
--- /dev/null
+++ b/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();
\ No newline at end of file
diff --git a/client/components/Reader/LoaderPage/LoaderPage.vue b/client/components/Reader/LoaderPage/LoaderPage.vue
index a9b6aaaf..57fe4852 100644
--- a/client/components/Reader/LoaderPage/LoaderPage.vue
+++ b/client/components/Reader/LoaderPage/LoaderPage.vue
@@ -11,8 +11,9 @@
Комментарии
@@ -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() {
diff --git a/client/components/Reader/Reader.vue b/client/components/Reader/Reader.vue
index efb8bd07..97740416 100644
--- a/client/components/Reader/Reader.vue
+++ b/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';
diff --git a/client/components/Reader/TextPage/TextPage.vue b/client/components/Reader/TextPage/TextPage.vue
new file mode 100644
index 00000000..f4bc4fb0
--- /dev/null
+++ b/client/components/Reader/TextPage/TextPage.vue
@@ -0,0 +1,34 @@
+
+
+ Text page
+
+
+
+
+
\ No newline at end of file
diff --git a/client/components/Reader/share/BookManager.js b/client/components/Reader/share/BookManager.js
new file mode 100644
index 00000000..98e368eb
--- /dev/null
+++ b/client/components/Reader/share/BookManager.js
@@ -0,0 +1,2 @@
+export default class BookManager {
+}
\ No newline at end of file
diff --git a/client/components/Reader/share/BookParser.js b/client/components/Reader/share/BookParser.js
new file mode 100644
index 00000000..e69de29b
diff --git a/client/components/Reader/share/easysax.js b/client/components/Reader/share/easysax.js
new file mode 100644
index 00000000..fa2444b6
--- /dev/null
+++ b/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() {}); //
+
+ 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('<') !== -1) {s = s.replace(/</g, '<');}
+ if (s.indexOf('>') !== -1) {s = s.replace(/>/g, '>');}
+ if (s.indexOf('"') !== -1) {s = s.replace(/"/g, '"');}
+
+ if (s.indexOf('&') !== -1) {
+ s = s.replace(/(\d+);|([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; //
+ 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) { // 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;
+ };
+ };
+};
diff --git a/client/share/utils.js b/client/share/utils.js
new file mode 100644
index 00000000..44179597
--- /dev/null
+++ b/client/share/utils.js
@@ -0,0 +1,3 @@
+export function sleep(ms) {
+ return new Promise(resolve => setTimeout(resolve, ms));
+}
diff --git a/client/store/modules/reader.js b/client/store/modules/reader.js
index 3292f65e..bb502025 100644
--- a/client/store/modules/reader.js
+++ b/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 {