diff --git a/server/src/main/kotlin/ir/armor/tachidesk/impl/Chapter.kt b/server/src/main/kotlin/ir/armor/tachidesk/impl/Chapter.kt index a50ffd10..6f8fb5ff 100644 --- a/server/src/main/kotlin/ir/armor/tachidesk/impl/Chapter.kt +++ b/server/src/main/kotlin/ir/armor/tachidesk/impl/Chapter.kt @@ -164,4 +164,26 @@ object Chapter { pageList.count() ) } + + fun modifyChapter(mangaId: Int, chapterIndex: Int, isRead: Boolean?, isBookmarked: Boolean?, markPrevRead: Boolean?, lastPageRead: Int?) { + transaction { + ChapterTable.update({ (ChapterTable.manga eq mangaId) and (ChapterTable.chapterIndex eq chapterIndex) }) { update -> + isRead?.also { + update[ChapterTable.isRead] = it + } + isBookmarked?.also { + update[ChapterTable.isBookmarked] = it + } + lastPageRead?.also { + update[ChapterTable.lastPageRead] = it + } + } + + markPrevRead?.let { + ChapterTable.update({ (ChapterTable.manga eq mangaId) and (ChapterTable.chapterIndex less chapterIndex) }) { + it[ChapterTable.isRead] = markPrevRead + } + } + } + } } diff --git a/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0001_Initial.kt b/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0001_Initial.kt index 59e743b2..bb8c976d 100644 --- a/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0001_Initial.kt +++ b/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0001_Initial.kt @@ -1,5 +1,12 @@ package ir.armor.tachidesk.model.database.migration +/* + * Copyright (C) Contributors to the Suwayomi project + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ + import eu.kanade.tachiyomi.source.model.SManga import ir.armor.tachidesk.model.database.migration.lib.Migration import org.jetbrains.exposed.dao.id.IdTable @@ -7,13 +14,6 @@ import org.jetbrains.exposed.dao.id.IntIdTable import org.jetbrains.exposed.sql.SchemaUtils import org.jetbrains.exposed.sql.transactions.transaction -/* - * Copyright (C) Contributors to the Suwayomi project - * - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ - class M0001_Initial : Migration() { private object ExtensionTable : IntIdTable() { val apkName = varchar("apk_name", 1024) @@ -100,6 +100,7 @@ class M0001_Initial : Migration() { val manga = reference("manga", ir.armor.tachidesk.model.database.table.MangaTable) } + /** initial migration, create all tables */ override fun run() { transaction { SchemaUtils.create( diff --git a/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0002_ChapterTableIndexRename.kt b/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0002_ChapterTableIndexRename.kt new file mode 100644 index 00000000..9c5eee0b --- /dev/null +++ b/server/src/main/kotlin/ir/armor/tachidesk/model/database/migration/M0002_ChapterTableIndexRename.kt @@ -0,0 +1,23 @@ +package ir.armor.tachidesk.model.database.migration + +/* + * Copyright (C) Contributors to the Suwayomi project + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ + +import ir.armor.tachidesk.model.database.migration.lib.Migration +import org.jetbrains.exposed.sql.transactions.TransactionManager +import org.jetbrains.exposed.sql.vendors.currentDialect + +class M0002_ChapterTableIndexRename : Migration() { + /** this migration renamed ChapterTable.NUMBER_IN_LIST to ChapterTable.INDEX */ + override fun run() { + with(TransactionManager.current()) { + exec("ALTER TABLE CHAPTER ALTER COLUMN NUMBER_IN_LIST RENAME TO INDEX") + commit() + currentDialect.resetCaches() + } + } +} diff --git a/server/src/main/kotlin/ir/armor/tachidesk/model/database/table/ChapterTable.kt b/server/src/main/kotlin/ir/armor/tachidesk/model/database/table/ChapterTable.kt index 30208ab9..674935c6 100644 --- a/server/src/main/kotlin/ir/armor/tachidesk/model/database/table/ChapterTable.kt +++ b/server/src/main/kotlin/ir/armor/tachidesk/model/database/table/ChapterTable.kt @@ -20,7 +20,8 @@ object ChapterTable : IntIdTable() { val isBookmarked = bool("bookmark").default(false) val lastPageRead = integer("last_page_read").default(0) - val chapterIndex = integer("number_in_list") + // index is reserved by a function + val chapterIndex = integer("index") val manga = reference("manga", MangaTable) } diff --git a/server/src/main/kotlin/ir/armor/tachidesk/model/dataclass/ChapterDataClass.kt b/server/src/main/kotlin/ir/armor/tachidesk/model/dataclass/ChapterDataClass.kt index 2f68dbb1..c0d824c8 100644 --- a/server/src/main/kotlin/ir/armor/tachidesk/model/dataclass/ChapterDataClass.kt +++ b/server/src/main/kotlin/ir/armor/tachidesk/model/dataclass/ChapterDataClass.kt @@ -25,7 +25,7 @@ data class ChapterDataClass( val lastPageRead: Int, /** this chapter's index, starts with 1 */ - val chapterIndex: Int? = null, + val index: Int? = null, /** total chapter count, used to calculate if there's a next and prev chapter */ val chapterCount: Int? = null, diff --git a/server/src/main/kotlin/ir/armor/tachidesk/server/JavalinSetup.kt b/server/src/main/kotlin/ir/armor/tachidesk/server/JavalinSetup.kt index 727c11b1..faded106 100644 --- a/server/src/main/kotlin/ir/armor/tachidesk/server/JavalinSetup.kt +++ b/server/src/main/kotlin/ir/armor/tachidesk/server/JavalinSetup.kt @@ -13,6 +13,7 @@ import ir.armor.tachidesk.impl.CategoryManga.getMangaCategories import ir.armor.tachidesk.impl.CategoryManga.removeMangaFromCategory import ir.armor.tachidesk.impl.Chapter.getChapter import ir.armor.tachidesk.impl.Chapter.getChapterList +import ir.armor.tachidesk.impl.Chapter.modifyChapter import ir.armor.tachidesk.impl.Extension.getExtensionIcon import ir.armor.tachidesk.impl.Extension.installExtension import ir.armor.tachidesk.impl.Extension.uninstallExtension @@ -142,7 +143,7 @@ object JavalinSetup { } // icon for extension named `apkName` - app.get("/api/v1/extension/icon/:apkName") { ctx -> + app.get("/api/v1/extension/icon/:apkName") { ctx -> // TODO: move to pkgName val apkName = ctx.pathParam("apkName") ctx.result( @@ -263,6 +264,22 @@ object JavalinSetup { ctx.json(future { getChapter(chapterIndex, mangaId) }) } + // used to modify a chapter's parameters + app.patch("/api/v1/manga/:mangaId/chapter/:chapterIndex") { ctx -> + val chapterIndex = ctx.pathParam("chapterIndex").toInt() + val mangaId = ctx.pathParam("mangaId").toInt() + + val read = ctx.formParam("read")?.toBoolean() + val bookmarked = ctx.formParam("bookmarked")?.toBoolean() + val markPrevRead = ctx.formParam("markPrevRead")?.toBoolean() + val lastPageRead = ctx.formParam("lastPageRead")?.toInt() + + modifyChapter(mangaId, chapterIndex, read, bookmarked, markPrevRead, lastPageRead) + + ctx.status(200) + } + + // get page at index "index" app.get("/api/v1/manga/:mangaId/chapter/:chapterIndex/page/:index") { ctx -> val mangaId = ctx.pathParam("mangaId").toInt() val chapterIndex = ctx.pathParam("chapterIndex").toInt() diff --git a/webUI/react/src/components/ChapterCard.tsx b/webUI/react/src/components/ChapterCard.tsx index 068d00e5..9877c06c 100644 --- a/webUI/react/src/components/ChapterCard.tsx +++ b/webUI/react/src/components/ChapterCard.tsx @@ -16,6 +16,7 @@ import Typography from '@material-ui/core/Typography'; import { Link, useHistory } from 'react-router-dom'; import Menu from '@material-ui/core/Menu'; import MenuItem from '@material-ui/core/MenuItem'; +import client from '../util/client'; const useStyles = makeStyles((theme) => ({ root: { @@ -65,13 +66,24 @@ export default function ChapterCard(props: IProps) { setAnchorEl(null); }; + const sendChange = (key: string, value: any) => { + console.log(`${key} -> ${value}`); + handleClose(); + + const formData = new FormData(); + formData.append(key, value); + client.patch(`/api/v1/manga/${chapter.mangaId}/chapter/${chapter.index}`, formData); + // .finally(() => triggerUpdate() + // ); + }; + return ( <>