mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 08:12:28 -04:00 
			
		
		
		
	* WebP Covers + Series Detail Enhancements (#1652) * Implemented save covers as webp. Reworked screen to provide more information up front about webp and what browsers can support it. * cleaned up pages to use compact numbering and made compact numbering expand into one decimal place (20.5K) * Fixed an issue with adding new device * If a book has an invalid language set, drop the language altogether rather than reading in a corrupted entry. * Ensure genres and tags render alphabetically. Improved support for partial volumes in Comic parser. * Ensure all people, tags, collections, and genres are in alphabetical order. * Moved some code to Extensions to clean up code. * More unit tests * Cleaned up release year filter css * Tweaked some code in all series to make bulk deletes cleaner on the UI. * Trying out want to read and unread count on series detail page * Added Want to Read button for series page to make it easy to see when something is in want to read list and toggle it. Added tooltips instead of title to buttons, but they don't style correctly. Added a continue point under cover image. * Code smells * Bump versions by dotnet-bump-version. * Fixed Series Relations Schema (#1654) * Bump loader-utils from 2.0.2 to 2.0.3 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.2 to 2.0.3. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.3/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.2...v2.0.3) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed is want to read coming back as a string and not working correctly. * Changed from to Continue to be more explicit * Added the first migration which exports data as a csv in temp/. This is the backup in case data is lost in the migration. * Note for later * Fixed the migration for the series relation so when deleting any series on any edge of the relationship, the SeriesRelation row deletes. * Change buttons back to titles on series detail page * Wrote the code to import relations from the backup. * Added an additional version check to avoid file io on migration. * Code cleanup Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Fresh Nightly Installs Work (#1659) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Removed a very old cgecj from Nov 2021 when data/appsettings was moved to config/ * Added some notes about migration * Removed a file that shouldn't have been there. Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Fixed GA (#1664) * Bump versions by dotnet-bump-version. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * [Experimental] Split Renderers - Double & Double (Manga) fixes (#1667) * Updated swiper and some packages for reported security issues * Fixed reading lists promotion not working * Refactor RenameFileForCopy to use iterative recursion, rather than functional. * Ensured that bookmarks are fetched and ordered by Created date. * Fixed a bug where bookmarks were coming back in the correct order, but due to filenames, would not sort correctly. * Default installs to Debug log level given errors users have and Debug not being too noisy * Added jumpbar to bookmarks page * Now added jumpbar to bookmarks * Refactored some code into pipes and added some debug messaging for prefetcher * Try loading next and prev chapter's first/last page to cache so it renders faster * Updated GetImage to do a bound check on max page. Fixed a critical bug in how manga reader updates image elements src to prefetch/load pages. I was not creating a new reference which broke Angular's ability to update DOM on changes. * Refactored the image setting code to use a single method which tries to use a cached image always. * Refactored code to use getPage which favors cache and simplifies image creation code * Started the work to split the canvas renderer into it's own component * Refactored a lot of common methods into a service for the reader to support the upcoming renderer split * Moved components to nested folder. Refactored more code to streamline image sending to child renderer. Added notes across the code to help streamline flow of data and who owns what. * Swapped out SQLite for Memory, but the one from hangfire. Added DisableConcurrentExecution on ProcessChange to avoid duplication when multiple threads execute at once. * Basic split right to left is working with canvas renderer * Left to right and right to left now work * Fixed a bug where pagesplitoption wasn't being updated when modifying menu * Canvas rendering still has a bug with switching between right to left -> left to right on the re-render, it will choose a bad state. All else works fine with it. * Updated canvas renderer to implement the ImageRenderer interface * Canvas renderer is done * Setup single renderer. Need to figure out how to share CSS between renderers and also share some global stuff, like image height. * Refactored code so that image-container is within the renderers themselves. Still broken in scaling, but working towards a solution. * Added double click to shortcut menu * Moved image containers within the renderers * Pushing up for Robbie * nothing new * Move common css to a single scss file * More css consolidation * Fixed a npe in isWideImage * Refactored page updates to renderers to include max pages. Rewrote most of renderer into observables. * Moved bookmark for second page to double renderer * Started hooking in double renderer renderPage() * Fixed height scaling, but now canvas renderer is broken again * Fixed a bug with canvas renderer not moving to next page. Streamlined the code for getting page amounts from the dfferent renderers * Added double click to bookmark for canvas * Stashing the code and taking a break * Nothing much, buffer is still broken * Got double renderer to render at least one page * Double renderer now has access to 5 images at any time, so it can make appropriate decisions on when to render double pages. * Fixed up double rendererer moving backward page calc * Forward logic seems to be working * Cleaned up dead code after testing * Moved a few loggers in folder watching to trace * Everything seems to work fine, time to do double manga renderer * Moved some css around and added the reverse double component * Only execute renderer's pipes when in the correct mode * Still working on double renderer * Fixed scaling issues on double * Updating double logic - Fixed: Fixed an issue where a second page would render when current page was wide. * Hooked up double renderer * Made changes but not sure if im making progress * double manga fixes * Claned some of robbies code * Fixing last page bug * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Fixed GA (#1664) * Bump versions by dotnet-bump-version. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Fixed typeahead and updated manga reader to new layout structure * Fixed book reader fonts lookups * Fixed up some build issues * Fixed a bad import of css image * Some cleanup and rewrote how we log out data. * Renderer can be null on first load when performing some work. * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Post merge cleanup * Again moving the file Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Basic Stats (#1673) * Refactored ResponseCache profiles into consts * Refactored code to use an extension method for getting user library ids. * Started server statistics, added a charting library, and added a table sort column (not finished) * Refactored code and have a fully working example of sortable headers. Still doesn't work with default sorting state, will work on that later. * Implemented file size, but it's too expensive, so commented out. * Added a migration to provide extension and length/size information in the DB to allow for faster stat apis. * Added the ability to force a library scan from library settings. * Refactored some apis to provide more of a file breakdown rather than just file size. * Working on visualization of file breakdown * Fixed the file breakdown visual * Fixed up 2 visualizations * Added back an api for member names, started work on top reads * Hooked up the other library types and username/days. * Preparing to remove top reads and refactor into Top users * Added LibraryId to AppUserProgress to help with complex lookups. * Added the new libraryId hook into some stats methods * Updated api methods to use libraryId for progress * More places where LibraryId is needed * Added some high level server stats * Got a ton done on server stats * Updated default theme (dark) to be the default root variables. This will allow user themes to override just what they want, rather than maintain their own css variables. * Implemented a monster query for top users by reading time. It's very slow and can be cleaned up likely. * Hooked up top reads. Code needs a big refactor. Handing off for Robbie treatment and I'll switch to User stats. * Implemented last 5 recently read series (broken) and added some basic css * Fixed recently read query * Cleanup the css a bit, Robbie we need you * More css love * Cleaned up DTOs that aren't needed anymore * Fixed top readers query * When calculating top readers, don't include read events where nothing is read (0 pages) * Hooked up the date into GetTopUsers * Hooked top readers up with days and refactored and cleaned up componets not used * Fixed up query * Started on a day by day breakdown, but going to take a break from stats. * Added a temp task to run some migration manually for stats to work * Ensure OPDS-PS uses new libraryId for progress reporting * Fixed a code smell * Adding some styling * adding more styles * Removed some debug stuff from user stats * Bump qs from 6.5.2 to 6.5.3 in /UI/Web Bumps [qs](https://github.com/ljharb/qs) from 6.5.2 to 6.5.3. - [Release notes](https://github.com/ljharb/qs/releases) - [Changelog](https://github.com/ljharb/qs/blob/main/CHANGELOG.md) - [Commits](https://github.com/ljharb/qs/compare/v6.5.2...v6.5.3) --- updated-dependencies: - dependency-name: qs dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Tweaked some code for bad data cases * Refactored a chapter lookup to remove un-needed Volume join in 5 places across the code. * API push Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Hooked up the API layer to be able to extract images from PDF again for Tachiyomi explicitly (#1686) * Bump versions by dotnet-bump-version. * OPDS Enhancements (#1687) * Bump express from 4.17.2 to 4.18.2 in /UI/Web Bumps [express](https://github.com/expressjs/express) from 4.17.2 to 4.18.2. - [Release notes](https://github.com/expressjs/express/releases) - [Changelog](https://github.com/expressjs/express/blob/master/History.md) - [Commits](https://github.com/expressjs/express/compare/4.17.2...4.18.2) --- updated-dependencies: - dependency-name: express dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Bump decode-uri-component from 0.2.0 to 0.2.2 in /UI/Web Bumps [decode-uri-component](https://github.com/SamVerschueren/decode-uri-component) from 0.2.0 to 0.2.2. - [Release notes](https://github.com/SamVerschueren/decode-uri-component/releases) - [Commits](https://github.com/SamVerschueren/decode-uri-component/compare/v0.2.0...v0.2.2) --- updated-dependencies: - dependency-name: decode-uri-component dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Bump qs and express in /UI/Web Bumps [qs](https://github.com/ljharb/qs) and [express](https://github.com/expressjs/express). These dependencies needed to be updated together. Updates `qs` from 6.5.3 to 6.11.0 - [Release notes](https://github.com/ljharb/qs/releases) - [Changelog](https://github.com/ljharb/qs/blob/main/CHANGELOG.md) - [Commits](https://github.com/ljharb/qs/compare/v6.5.3...v6.11.0) Updates `express` from 4.17.2 to 4.18.2 - [Release notes](https://github.com/expressjs/express/releases) - [Changelog](https://github.com/expressjs/express/blob/master/History.md) - [Commits](https://github.com/expressjs/express/compare/4.17.2...4.18.2) --- updated-dependencies: - dependency-name: qs dependency-type: indirect - dependency-name: express dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Added genre and authors to Series level, added summary to volume and chapter level. Force order on reading list title as Chunky enforces their own sort order and doesn't respect the spec. * Moved all the reading list formatting logic to the backend. This allows us to re-use the UI logic for OPDS streams. * Fixed a broken unit test * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * Epub Table of Generation fixes for Sigil (#1689) * Fixed generating table of contents where key lookup could fail with how Sigil packs the epubs. * Tweaked Kavita's fallback ToC generation (when one doesn't exist in the epub) to also use CoalesceKey. * Code smells * Bump versions by dotnet-bump-version. * File Dimension API (#1690) * Implemented an api for getting file dimensions for a given chapter. This is for CDisplayEx integration. This might be usable in Double Renderer. * Added the cached filename for new API * Bump versions by dotnet-bump-version. * Send Non books to your Devices (#1691) * Only restrict non-epub/pdf for Kindle devices on Send To. * Removed restriction to email non-epub/pdfs to devices. * Bump versions by dotnet-bump-version. * Misc UI Tweaks (#1692) * Added a timeAgo pipe which shows live updates for a few areas. * Fixed some wording on stats page. Changed Total People count to just work on distinct names and not count multiple for different roles. * Tweaked the compact number so it only shows one decimal * Fixed a bug * Bump versions by dotnet-bump-version. * Reader Refactor Part 2 (#1694) * Updated swiper and some packages for reported security issues * Fixed reading lists promotion not working * Refactor RenameFileForCopy to use iterative recursion, rather than functional. * Ensured that bookmarks are fetched and ordered by Created date. * Fixed a bug where bookmarks were coming back in the correct order, but due to filenames, would not sort correctly. * Default installs to Debug log level given errors users have and Debug not being too noisy * Added jumpbar to bookmarks page * Now added jumpbar to bookmarks * Refactored some code into pipes and added some debug messaging for prefetcher * Try loading next and prev chapter's first/last page to cache so it renders faster * Updated GetImage to do a bound check on max page. Fixed a critical bug in how manga reader updates image elements src to prefetch/load pages. I was not creating a new reference which broke Angular's ability to update DOM on changes. * Refactored the image setting code to use a single method which tries to use a cached image always. * Refactored code to use getPage which favors cache and simplifies image creation code * Started the work to split the canvas renderer into it's own component * Refactored a lot of common methods into a service for the reader to support the upcoming renderer split * Moved components to nested folder. Refactored more code to streamline image sending to child renderer. Added notes across the code to help streamline flow of data and who owns what. * Swapped out SQLite for Memory, but the one from hangfire. Added DisableConcurrentExecution on ProcessChange to avoid duplication when multiple threads execute at once. * Basic split right to left is working with canvas renderer * Left to right and right to left now work * Fixed a bug where pagesplitoption wasn't being updated when modifying menu * Canvas rendering still has a bug with switching between right to left -> left to right on the re-render, it will choose a bad state. All else works fine with it. * Updated canvas renderer to implement the ImageRenderer interface * Canvas renderer is done * Setup single renderer. Need to figure out how to share CSS between renderers and also share some global stuff, like image height. * Refactored code so that image-container is within the renderers themselves. Still broken in scaling, but working towards a solution. * Added double click to shortcut menu * Moved image containers within the renderers * Pushing up for Robbie * nothing new * Move common css to a single scss file * More css consolidation * Fixed a npe in isWideImage * Refactored page updates to renderers to include max pages. Rewrote most of renderer into observables. * Moved bookmark for second page to double renderer * Started hooking in double renderer renderPage() * Fixed height scaling, but now canvas renderer is broken again * Fixed a bug with canvas renderer not moving to next page. Streamlined the code for getting page amounts from the dfferent renderers * Added double click to bookmark for canvas * Stashing the code and taking a break * Nothing much, buffer is still broken * Got double renderer to render at least one page * Double renderer now has access to 5 images at any time, so it can make appropriate decisions on when to render double pages. * Fixed up double rendererer moving backward page calc * Forward logic seems to be working * Cleaned up dead code after testing * Moved a few loggers in folder watching to trace * Everything seems to work fine, time to do double manga renderer * Moved some css around and added the reverse double component * Only execute renderer's pipes when in the correct mode * Still working on double renderer * Fixed scaling issues on double * Updating double logic - Fixed: Fixed an issue where a second page would render when current page was wide. * Hooked up double renderer * Made changes but not sure if im making progress * double manga fixes * Claned some of robbies code * Fixing last page bug * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Fixed GA (#1664) * Bump versions by dotnet-bump-version. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Fixed typeahead and updated manga reader to new layout structure * Fixed book reader fonts lookups * Fixed up some build issues * Fixed a bad import of css image * Some cleanup and rewrote how we log out data. * Renderer can be null on first load when performing some work. * Library Settings Modal + New Library Settings (#1660) * Bump loader-utils from 2.0.3 to 2.0.4 in /UI/Web Bumps [loader-utils](https://github.com/webpack/loader-utils) from 2.0.3 to 2.0.4. - [Release notes](https://github.com/webpack/loader-utils/releases) - [Changelog](https://github.com/webpack/loader-utils/blob/v2.0.4/CHANGELOG.md) - [Commits](https://github.com/webpack/loader-utils/compare/v2.0.3...v2.0.4) --- updated-dependencies: - dependency-name: loader-utils dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Fixed want to read button on series detail not performing the correct action * Started the library settings. Added ability to update a cover image for a library. Updated backup db to also copy reading list (and now library) cover images. * Integrated Edit Library into new settings (not tested) and hooked up a wizard-like flow for new library. * Fixed a missing update event in backend when updating a library. * Disable Save when form invalid. Do inline validation on Library name when user types to ensure the name is valid. * Trim library names before you check anything * General code cleanup * Implemented advanced settings for library (include in dashboard, search, recommended) and ability to turn off folder watching for individual libraries. Refactored some code to streamline perf in some flows. * Removed old components replaced with new modal * Code smells Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * UX Alignment and bugfixes (#1663) * Refactored the design of reading list page to follow more in line with list view. Added release date on the reading list items, if it's set in underlying chapter. Fixed a bug where reordering the list items could sometimes not update correctly with drag and drop. * Removed a bug marker that I just fixed * When generating library covers, make them much smaller as they are only ever icons. * Fixed library settings not showing the correct image. * Fixed a bug where duplicate collection tags could be created. Fixed a bug where collection tag normalized title was being set to uppercase. Redesigned the edit collection tag modal to align with new library settings and provide inline name checks. * Updated edit reading list modal to align with new library settings modal pattern. Refactored the backend to ensure it flows correctly without allowing duplicate names. Don't show Continue point on series detail if the whole series is read. * Added some more unit tests around continue point * Fixed a bug on series detail when bulk selecting between volume and chapters, the code which determines which chapters are selected didn't take into account mixed layout for Storyline tab. * Refactored to generate an OpenAPI spec at root of Kavita. This will be loaded by a new API site for easy hosting. Deprecated EnableSwaggerUi preference as after validation new system works, this will be removed and instances can use our hosting to hit their server (or run a debug build). * Test GA * Reverted GA and instead do it in the build step. This will just force developers to commit it in. * GA please work * Removed redundant steps from test since build already does it. * Try another GA * Moved all test actions into initial build step, which should drastically cut down on time. Only run sonar if the secret is present (so not for forks). Updated build requirements for develop and stable docker pushes. * Fixed env variable * Okay not possible to do secrets in if statement * Fixed the build step to output the openapi.json where it's expected. * Applied new _components layout structure to Kavita. All except manga as there is an open PR that drastically changes that module. (#1666) * Post merge cleanup * Again moving the file * Fixed an issue with switching to double renderer and the image not loading for cover image. * Fixed double manga last page repeating twice * Added ability to quickly save a few settings to user preferences from manga reader * Fixed up some success messaging * Single image and canvas could stack, last page on double wouldn't render. * Stashing code, want to work on something else * Suppress a concurrency issue when opening a fresh chapter to read. * Refactored a function into a pipe * Took care of one TODO * Tightened up the logic around single renderer handling fit to screen images. * Added some code to see how long api takes on average. * First pass integration of page dimensions into single renderer and base code * Canvas renderer pass for new page dimensions * On time left, don't use the word left again * Moved the page dimension code into manga service to make it seemless * Hooked in a replacement for image based isWide * Canvas renderer is working again * Double renderer now follows how Komga does it to keep it simple. * Double renderer is working really well so far. * don't use nbsp * Added response caching to file-dimensions and chapter info api * Allow chapter info to send back file dimensions optionally * Fixed an issue with dimensions api locking files on Windows * Refactored all code to use isWidePage * More fixes and cleanup * More double reverse logic * Recently Read stats page will allow you to click the items. Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * More Reader Fixes (#1696) * Fixed resizing or layout changes causing page change on double reader * Implemented the debug log pattern on double renderers. Fixed a case when navigation backwards and showing only one page. Updated so go to page or slider update will handle selecting the right page number for pair display. * All Spread cases for double working * Cleanup dead code * Ensure we can jump to last page * Bump versions by dotnet-bump-version. * Reading History (#1699) * Added new stat graph for pages read over time for all users. * Switched to reading events rather than pages read to get a better scale * Changed query to use Created date as LastModified wont work since I just did a migration on all rows. * Small cleanup on graph * Read by day completed and ready for user stats page. * Changed the initial stat report to be in 1 day, to avoid people trying and ditching the software from muddying up the stats. * Cleaned up stats page such that stats around series show their image and tweaked some layout and wordings * Fixed recently read order * Put read history on user profile * Final cleanup, Robbie needs to do a CSS pass before release. * Bump versions by dotnet-bump-version. * Performance Improvements and Some Polish (#1702) * Auto scale reading timeline * Added benchmarks for SharpImage and NetVips. When an epub has a malformed page, catch the error and present it better to the user. * Added a hint for an upcoming feature * Slightly sped up word count for epubs * Added one more test to reflect actual code. * Some light cleanup * Use compact number for stat lists * Fixed brightness being broken on manga reader * Replaced CoverToWebP SharpImage version with NetVips which is MUCH lighter on memory and CPU. * Added last modified on the progress dto for CdDisplayEx. * Code cleanup * Forgot one cleanup * Bump versions by dotnet-bump-version. * Holiday Changes (#1706) * Fixed a bug on bookmark mode not finding correct image for prefetcher. * Fixed up the edit series relationship modal on tablet viewports. * On double page mode, only bookmark 1 page if only 1 pages is renderered on screen. * Added percentage read of a given library and average hours read per week to user stats. * Fixed a bug in the reader with paging in bookmark mode * Added a "This Week" option to top readers history * Added date ranges for reading time. Added dates that don't have anything, but might remove. * On phone, when applying a metadata filter, when clicking apply, collapse the filter automatically. * Disable jump bar and the resuming from last spot when a custom sort is applied. * Ensure all Regex.Replace or Matches have timeouts set * Bump versions by dotnet-bump-version. * First PR of the new year (#1717) * Fixed a bug on bookmark mode not finding correct image for prefetcher. * Fixed up the edit series relationship modal on tablet viewports. * On double page mode, only bookmark 1 page if only 1 pages is renderered on screen. * Added percentage read of a given library and average hours read per week to user stats. * Fixed a bug in the reader with paging in bookmark mode * Added a "This Week" option to top readers history * Added date ranges for reading time. Added dates that don't have anything, but might remove. * On phone, when applying a metadata filter, when clicking apply, collapse the filter automatically. * Disable jump bar and the resuming from last spot when a custom sort is applied. * Ensure all Regex.Replace or Matches have timeouts set * Fixed a long standing bug where fit to height on tablets wouldn't center the image * Streamlined url parsing to be more reliable * Reduced an additional db query in chapter info. * Added a missing task to convert covers to webP and added messaging to help the user understand to run it after modifying the setting. * Changed OPDS to be enabled by default for new installs. This should reduce issues with users being confused about it before it's enabled. * When there are multiple files for a chapter, show a count card on the series detail to help user understand duplicates exist. Made the unread badge smaller to avoid collision. * Added Word Count to user stats and wired up average reading per week. * Fixed word count failing on some epubs * Removed some debug code * Don't give more information than is necessary about file paths for page dimensions. * Fixed a bug where pagination area would be too small when the book's content was less that height on default mode. * Updated Default layout mode to Scroll for books. * Added bytes in the UI and at an API layer for CDisplayEx * Don't log health checks to logs at all. * Changed Word Count to Length to match the way pages work * Made reading time more clear when min hours is 0 * Apply more aggressive coalescing when remapping bad metadata keys for epubs. * Changed the amount of padding between icon and text for side nav item. * Fixed a NPE on book reader (harmless) * Fixed an ordering issue where Volume 1 was a single file but also tagged as Chapter 1 and Volume 2 was Chapter 0. Thus Volume 2 was being selected for continue point when Volume 1 should have been. * When clicking on an activity stream header from dashboard, show the title on the resulting page. * Removed a property that can't be animated * Fixed a typeahead typescript issue * Added Size into Series Info and Added some tooltip and spacing changes to better explain some fields. * Added size for volume drawers and cleaned up some date edge case handling * Fixed an annoying bug where when on mobile opening a view with a metadata filter, Kavita would open the filter automatically. * Bump versions by dotnet-bump-version. * Quick fix for Double Renderer (#1719) * Disable emulate comic book when on single page reader * Fixed a regression where double page renderer wouldn't layout the images correctly * Bump versions by dotnet-bump-version. * Feature/stats finishoff (#1720) * Added ability to click on genres, tags, and people to view all items in a modal. * Made it so we can click and open a filtered search from generic list * Fixed broken epub pagination area due to a typo in a query selector * Added day breakdown, wrapping up stats * Bump versions by dotnet-bump-version. * Docker nonroot (#1650) * Added PUID, PGID and KAVITAUSER variable support in entrypoint.sh * Update the setting of ownership to avoid changing library files * Default to run as root, using user kavita if alternate UID/GID are provided * Only chown config folder and only if needed * Revert chmod on Kavita Co-authored-by: Muggz <mug@passw0rd.org> * Bump versions by dotnet-bump-version. * Manga Reader Work (#1729) * Instead of augmenting prefetcher to move across chapter bounds, let's try to instead just load 5 images (which the browser will cache) from next/prev so when it loads, it's much faster. * Trialing loading next/prev chapters 5 pages to have better next page loading experience. * Tweaked GetChapterInfo API to actually apply conditional includeDimensions parameter. * added a basic language file for upcoming work * Moved the bottom menu up a bit for iOS devices with handlebars. * Fixed fit to width on phones still having a horizontal scrollbar * Fixed a bug where there is extra space under the image when fit to width and on a phone due to pagination going to far. * Changed which variable we use for right pagination calculation * Fixing fit to height - Fixing height calc to account for horizontal scroll bar height. * Added a comment for the height scrollbar fix * Adding screenfull package # Added: - Added screenfull package to handle cross-platform browser fullscreen code # Removed: - Removed custom fullscreen code * Fixed a bug where switching from webtoon reader to other layout modes wouldn't render anything. Webtoon continuous scroll down is now broken. * Fixed it back to how it was and all is good. Need to call detectChanges explicitly. * Removed an additional undeeded save progress call on loadPage * Laid out the test case to move the page snapping to the backend with full unit tests. Current code is broken just like UI layer. * Refactored the snap points into the backend and ensure that it works correctly. * Fixed a broken unit test * Filter out spammy hubs/messages calls in the logs * Swallow all noisy messages that are from RequestLoggingMiddleware when the log level is on Information or above. * Added a common loading component to the app. Have yet to refactor all screens to use this. * Bump json5 from 2.2.0 to 2.2.3 in /UI/Web Bumps [json5](https://github.com/json5/json5) from 2.2.0 to 2.2.3. - [Release notes](https://github.com/json5/json5/releases) - [Changelog](https://github.com/json5/json5/blob/main/CHANGELOG.md) - [Commits](https://github.com/json5/json5/compare/v2.2.0...v2.2.3) --- updated-dependencies: - dependency-name: json5 dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Alrigned all the loading messages and styles throughout the app * Webtoon reader will use max width of all images to ensure images align well. * On Original scaling mode, users can use the keyboard to scroll around the images without pagination kicking off. * Removed console logs * Fixed a public vs private issue * Fixed an issue around some cached files getting locked due to NetVips holding them during file size calculations. Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump versions by dotnet-bump-version. * [Manga Reader] Swipe Support (#1735) * Fixed a loading indicator that is always on * Started to add swipe directive * Implemented the ability to swipe to navigate pages in manga reader. * Swipe to paginate seems to be working reliably * Removed a bunch of junk from csproj and added a debug menu for testing on phone to smooth out experience. * Fixed a bug where reading list detail wouldn't render the set image of the reading list. * Added some instructions and code to allow connecting to dev instance easier. * Fixed up paging with keyboard where to ensure that when we hit the end of the scroll, we don't go to the next page instantly, but rather make the user press the key once more. * Fixed reading list image not properly renderering on reading list detail page. * Solved the swiping bug, need to play with threshold again. * Swipe is now working. Need to decide if I'm going to support reversing the direction with reading direction. * Hooked up swipe with reading direction code * Cleaned up some direction code to align to a new enum * Feature complete * Bump versions by dotnet-bump-version. * Fix pagination in Manga reader from last PR (#1736) * Added a new Double (No Cover) rendering mode which always has first 2 pages together unless wide. * Removed layout mode for build * Bump versions by dotnet-bump-version. * Better Themes, Stats, and bugfixes (#1740) * Fixed a bug where when clicking on a series rating for first time, the rating wasn't populating in the modal. * Fixed a bug on Scroll mode with immersive mode, the bottom bar could clip with the book body. * Cleanup some uses of var * Refactored text as json into a type so I don't have to copy/paste everywhere * Theme styles now override the defaults and theme owners no longer need to maintain all the variables themselves. Themes can now override the color of the header on mobile devices via --theme-color and Kavita will now update both theme color as well as color scheme. * Fixed a bug where last active on user stats wasn't for the particular user. * Added a more accurate word count calculation and the ability to see the word counts year over year. * Added a new table for long term statistics, like number of files over the years. No views are present for this data, I will add them later. * Bump versions by dotnet-bump-version. * Swipe Issues (#1745) * Updated theme support to be able to customize the tile color dynamically from a theme via --tile-color. In addition, --theme-color will update apple-mobile-web-app-status-bar-style as well as the non-apple variants * Removed --manga-reader-bg-color as it wasn't used anywhere. Fixed double pagination on swipe. * Cleaned up some dead threshold code for swipe. * Started refactoring tests to use an abstract test class. Stopping because I should do on the .net 7 branch to avoid large merge conflicts. Tests need to be re-designed so they can run in parallel. * Fixed a bug in reading lists where when deleting an item, order could be miscalculated. * Started adding new information for stat service. Refactored time spent reading to be more accurate by taking average time against how much of the chapter the user has read. * Hooked up total time reading at server stat level. Don't show fancy graphs on mobile. * Added new stats for v0.7 * Added a test for Clearing want to read * Fixed a few tests that weren't resetting state between runs * Fixed some broken unit tests * Ensure all Series queries sort by a case invariant string. * Added more aggressive caching of images. This will result in a min delay on pages after a cover is changed. * Fixed a bug where if during new word count calculation, new word count is zero, restoring the old count wasn't working. * Cleaned up some of the code for getting time estimates * Fixed a bug where triggering swipe right wasn't working when there was no scroll * Delete the temp folder for creating a download after a full zip is created. * Bump versions by dotnet-bump-version. * Stat hotfix (#1748) * Fixed a bug where a divide by 0 could occur * Email change now requires a password * Bump versions by dotnet-bump-version. * Holiday Bugfixes (#1762) * Don't show "not much going on" when we are actively downloading * Swipe to paginate is now behind a flag in the user preferences. * Added a new server setting for host name, if the server sits behind a reverse proxy. If this is set, email link generation will use it and will not perform any checks on accessibility (thus email will always send) * Refactored the code that checks if the server is accessible to check if host name is set, and thus return rue if so. * Added back the system drawing library for markdown parsing. * Fixed a validation error * Fixed a bug where folder watching could get re-triggered when it was disabled at a server level. * Made the manga reader loader absolute positioned for better visibility * Indentation * Bump versions by dotnet-bump-version. * Angular 15 (#1764) * Updated ngx-virtual-scroller * Removed the karma test config as it's breaking migration * Reverted to pre angular 15 * Upgraded packages and reverted target to ES6 for older devices * It's broken. Need to also find the safari version for old Ipads * Fixes some code in default pipe and many updates to packages. Removed support for old iOS versions as it restricted Kavita from using newer features. Build still broken. * More progress in getting build working on Angular 15. Removed polyfills.ts for new angular config * Remove all.css for icons and use scss instead * Removed stuff that isn't needed * Migrated extended linting to eslint, ran on project and updated issues. Removed a duplicate component that did nothing. Fixed a few places where lifecycle hooks werent being called as interface wasn't implemented. * App builds correctly. Source maps are still needed. * Fixed source maps and removed more testing stuff. I will re-add later in another release when I figure out how to properly tackle dependencies on backend. * Reverted back to old source map definition * Bump versions by dotnet-bump-version. * Angular 15 (#1765) * Refactored some code in BookService to make the code easier to understand * More lint fixes * Use npm ci for installs in pipeline * Fixed build system again by deleting nodejs. New build system uses package-lcok going forward. * Bump versions by dotnet-bump-version. * [skip ci] Misc stuff (#1766) * Refactored some code in BookService to make the code easier to understand * More lint fixes * Use npm ci for installs in pipeline * Fixed build system again by deleting nodejs. New build system uses package-lcok going forward. * Added a test case for Reading Time Estimation calculations * Some cleanup * Added even more testing to try and get scare's issue captured. * Automatic Collection Creation (#1768) * Made the unread badges slightly smaller and rounded on top right. * A bit more tweaks on the not read badges. Looking really nice now. * In order to start the work on managing collections from ScanLoop, I needed to refactor collection apis into the service layer and add unit tests. Removed ToUpper Normalization for new tags. * Hooked up ability to auto generate collections from SeriesGroup metadata tag. * Bump versions by dotnet-bump-version. * Auto Collection Bugfixes (#1769) * SeriesGroup tag can now have comma separated value to allow a series to be a part of multiple collections. * Added a missing unit test * Refactored how collection tags are created to work in the scan loop reliably. * Added a unit test for RemoveTagsWithoutSeries * Fixed a bug in reading list title generation to avoid Volume 0 if the underlying file had a title set. Fixed a misconfigured unit test. * Bump versions by dotnet-bump-version. * Scanner Performance Improvements (#1774) * Refactored the Genre code to be faster and used a dictonary to avoid some lookups. May fix the rare foreign constraint issue. * Refactored tag to the same implementation as Genre. Ensure when grabbing tags from ComicInfo, we normalize and throw out duplicates. * Removed an internal "external" field that was planned for Genres and Tags, but now with new plugin architecture, not needed. * Bump versions by dotnet-bump-version. * Spelling, grammar, and related consistency improvements (#1756) * Spelling, grammar, and word structure improvements * Email service text reworded to account for the Host Name feature * Bump versions by dotnet-bump-version. * Stat Polish (#1775) * SeriesGroup tag can now have comma separated value to allow a series to be a part of multiple collections. * Added a missing unit test * Refactored how collection tags are created to work in the scan loop reliably. * Added a unit test for RemoveTagsWithoutSeries * Fixed a bug in reading list title generation to avoid Volume 0 if the underlying file had a title set. Fixed a misconfigured unit test. * On User stats page, don't show the user selector on reading history, despite if youre an admin. Cleaned up how we show days with 0 reading events to be more clear. * Refactored the name of a component to reflect what it does * Removed plugin not using * Fix an issue where coalescing a key in epub might have multiple html files ending with the key. In this case, let's take the first. * Added PikaPods to the Readme * Tried to fix layout shift for charts, but need Robbie's help * Chart styling # Added: - Added: Added styling to force charts into their respective containers. # Removed: - Removed: Removed code blocking charts from being visible on mobile. * Merge conflict --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> * Bump versions by dotnet-bump-version. * UTC Dates + CDisplayEx API Enhancements (#1781) * Introduced a new claim on the Token to get UserId as well as Username, thus allowing for many places of reduced DB calls. All users will need to reauthenticate. Introduced UTC Dates throughout the application, they are not exposed in all DTOs, that will come later when we fully switch over. For now, Utc dates will be updated along side timezone specific dates. Refactored get-progress/progress api to be 50% faster by reducing how much data is loaded from the query. * Speed up the following apis: collection/search, download/bookmarks, reader/bookmark-info, recommended/quick-reads, recommended/quick-catchup-reads, recommended/highly-rated, recommended/more-in, recommended/rediscover, want-to-read/ * Added a migration to sync all dates with their new UTC counterpart. * Added LastReadingProgressUtc onto ChapterDto for some browsing apis, but not all. Added LastReadingProgressUtc to reading list items. Refactored the migration to run raw SQL which is much faster. * Added LastReadingProgressUtc onto ChapterDto for some browsing apis, but not all. Added LastReadingProgressUtc to reading list items. Refactored the migration to run raw SQL which is much faster. * Fixed the unit tests * Fixed an issue with auto mapper which was causing progress page number to not get sent to UI * series/volume has chapter last reading progress * Added filesize and library name on reading list item dto for CDisplayEx. * Some minor code cleanup * Forgot to fill a field * Bump versions by dotnet-bump-version. * Reading List Fixes (#1784) * Add ability to save readinglist comicinfo fields in Chapter. * Added the appropriate fields and migration for Reading List generation. * Started the reading list code * Started building out the CBL import code with some initial unit tests. * Fixed first unit test * Started refactoring control code into services and writing unit tests for ReadingLists. Found a logic issue around reading list title between create/update. Will be corrected in this branch with unit tests. * Can't figure out how to mock UserManager, so had to uncomment a few tests. * Tooltip for total pages read shows the full number * Tweaked the math a bit for average reading per week. * Fixed up the reading list unit tests. Fixed an issue where when inserting chapters into a blank reading list, the initial reading list item would have an order of 1 instead of 0. * Cleaned up the code to allow the reading list code to be localized easily and fixed up a bug in last PR. * Fixed a sorting issue on reading activity * Tweaked the code around reading list actionables not showing due to some weird filter. * Fixed edit library settings not opening on library detail page * Fixed a bug where reading activity dates would be out of order due to a bug in how charts works. A temp hack has been added. * Disable promotion in edit reading list modal since non-admins can (and should have) been able to use it. * Fixed a bug where non-admins couldn't update their OWN reading lists. Made uploading a cover image for readinglists now check against the user's reading list access to allow non-admin's to set images. * Fixed an issue introduced earlier in PR where adding chapters to reading list could cause order to get skewed. * Fixed another regression from earlier commit * Hooked in Import CBL flow. No functionality yet. * Code is a mess. Shifting how the whole import process is going to be done. Commiting so I can pivot drastically. * Very rough code for first step is done. * Ui has started, I've run out of steam for this feature. * Cleaned up the UI code a bit to make the step tracker nature easier without a dedicated component. * Much flow implementation and tweaking to how validation checks and what is sent back. * Removed import via cbl code as it's not done. Pushing to next release. * Bump versions by dotnet-bump-version. * Allow changing listening ip addresses (#1713) * Allow changing listening ip address * Use Json serialize for appsettings.config saving * BOM * IP Address validation * ip address reset * ValidIpAddress regex * Bump versions by dotnet-bump-version. * Release Testing Time (#1785) * Fixed a bug with getting continue point where there was a single volume unread and a later volume with chapters inside it, the chapters were being picked. * Fixed a bug where resuming from jump key wasn't working (develop) * Cleaned up the spacing * Bump versions by dotnet-bump-version. * Release Testing Bugs (#1790) * Stop showing loading indicator when no next/prev chapter * Fixed a bug where manage collections wasn't named correctly in UI. * Slight tweaks on email flow * Bump versions by dotnet-bump-version. * Release Testing Part 2 (#1794) * Stop showing loading indicator when no next/prev chapter * Fixed a bug where manage collections wasn't named correctly in UI. * Slight tweaks on email flow * Fixed a bug where we were grabbing wrong property for book layout mode * Fixed an issue where pagination area wasn't properly spanning window on different scaling modes. * Fixed a bug where right pagination area wasn't sticking to the right hand side on original scaling * Added a note from reading an epub3 * Reworked some of the readme * Bump versions by dotnet-bump-version. * Small Build Fix (#1795) * Stop showing loading indicator when no next/prev chapter * Fixed a bug where manage collections wasn't named correctly in UI. * Slight tweaks on email flow * Fixed a bug where we were grabbing wrong property for book layout mode * Fixed an issue where pagination area wasn't properly spanning window on different scaling modes. * Fixed a bug where right pagination area wasn't sticking to the right hand side on original scaling * Added a note from reading an epub3 * Reworked some of the readme * Changed the build to ci * Bump versions by dotnet-bump-version. * Final Release Testing (#1796) * Fix some wording * Fixed up stats to have total info on hover * Fixed up a stat card not having clickable hint * Bump versions by dotnet-bump-version. * v0.7 Release * Version bump --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: Mike <github@emailisgood.com> Co-authored-by: Muggz <mug@passw0rd.org> Co-authored-by: Domenic Fiore <DomenicF@users.noreply.github.com> Co-authored-by: Kupferhirn <kupferhirn@brokensoft.net>
		
			
				
	
	
		
			1674 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1674 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| // <auto-generated />
 | |
| using System;
 | |
| using API.Data;
 | |
| using Microsoft.EntityFrameworkCore;
 | |
| using Microsoft.EntityFrameworkCore.Infrastructure;
 | |
| using Microsoft.EntityFrameworkCore.Migrations;
 | |
| using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
 | |
| 
 | |
| #nullable disable
 | |
| 
 | |
| namespace API.Data.Migrations
 | |
| {
 | |
|     [DbContext(typeof(DataContext))]
 | |
|     [Migration("20221115021908_SeriesRelationChange")]
 | |
|     partial class SeriesRelationChange
 | |
|     {
 | |
|         protected override void BuildTargetModel(ModelBuilder modelBuilder)
 | |
|         {
 | |
| #pragma warning disable 612, 618
 | |
|             modelBuilder.HasAnnotation("ProductVersion", "6.0.10");
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppRole", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("ConcurrencyStamp")
 | |
|                         .IsConcurrencyToken()
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasMaxLength(256)
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedName")
 | |
|                         .HasMaxLength(256)
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("NormalizedName")
 | |
|                         .IsUnique()
 | |
|                         .HasDatabaseName("RoleNameIndex");
 | |
| 
 | |
|                     b.ToTable("AspNetRoles", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUser", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AccessFailedCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AgeRestriction")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("AgeRestrictionIncludeUnknowns")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("ApiKey")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("ConcurrencyStamp")
 | |
|                         .IsConcurrencyToken()
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("ConfirmationToken")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Email")
 | |
|                         .HasMaxLength(256)
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("EmailConfirmed")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("LastActive")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("LockoutEnabled")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTimeOffset?>("LockoutEnd")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedEmail")
 | |
|                         .HasMaxLength(256)
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedUserName")
 | |
|                         .HasMaxLength(256)
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("PasswordHash")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("PhoneNumber")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("PhoneNumberConfirmed")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<uint>("RowVersion")
 | |
|                         .IsConcurrencyToken()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("SecurityStamp")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("TwoFactorEnabled")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("UserName")
 | |
|                         .HasMaxLength(256)
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("NormalizedEmail")
 | |
|                         .HasDatabaseName("EmailIndex");
 | |
| 
 | |
|                     b.HasIndex("NormalizedUserName")
 | |
|                         .IsUnique()
 | |
|                         .HasDatabaseName("UserNameIndex");
 | |
| 
 | |
|                     b.ToTable("AspNetUsers", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserBookmark", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ChapterId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("FileName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Page")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("VolumeId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId");
 | |
| 
 | |
|                     b.ToTable("AppUserBookmark");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserPreferences", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("AutoCloseMenu")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("BackgroundColor")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("TEXT")
 | |
|                         .HasDefaultValue("#000000");
 | |
| 
 | |
|                     b.Property<bool>("BlurUnreadSummaries")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("BookReaderFontFamily")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("BookReaderFontSize")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("BookReaderImmersiveMode")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("BookReaderLayoutMode")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("BookReaderLineSpacing")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("BookReaderMargin")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("BookReaderReadingDirection")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("BookReaderTapToPaginate")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("BookThemeName")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("TEXT")
 | |
|                         .HasDefaultValue("Dark");
 | |
| 
 | |
|                     b.Property<int>("GlobalPageLayoutMode")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER")
 | |
|                         .HasDefaultValue(0);
 | |
| 
 | |
|                     b.Property<int>("LayoutMode")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("NoTransitions")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("PageSplitOption")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("PromptForDownloadSize")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ReaderMode")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ReadingDirection")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ScalingOption")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("ShowScreenHints")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int?>("ThemeId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId")
 | |
|                         .IsUnique();
 | |
| 
 | |
|                     b.HasIndex("ThemeId");
 | |
| 
 | |
|                     b.ToTable("AppUserPreferences");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserProgress", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("BookScrollId")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("ChapterId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("PagesRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("VolumeId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId");
 | |
| 
 | |
|                     b.HasIndex("SeriesId");
 | |
| 
 | |
|                     b.ToTable("AppUserProgresses");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserRating", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("Rating")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Review")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId");
 | |
| 
 | |
|                     b.HasIndex("SeriesId");
 | |
| 
 | |
|                     b.ToTable("AppUserRating");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserRole", b =>
 | |
|                 {
 | |
|                     b.Property<int>("UserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("RoleId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("UserId", "RoleId");
 | |
| 
 | |
|                     b.HasIndex("RoleId");
 | |
| 
 | |
|                     b.ToTable("AspNetUserRoles", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Chapter", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AgeRating")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AvgHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("Count")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("CoverImage")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("CoverImageLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("IsSpecial")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Language")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("MaxHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("MinHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Number")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Pages")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Range")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("ReleaseDate")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Summary")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Title")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("TitleName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("TotalCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("VolumeId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<long>("WordCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("VolumeId");
 | |
| 
 | |
|                     b.ToTable("Chapter");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.CollectionTag", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("CoverImage")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("CoverImageLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("NormalizedTitle")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("Promoted")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<uint>("RowVersion")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Summary")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Title")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("Id", "Promoted")
 | |
|                         .IsUnique();
 | |
| 
 | |
|                     b.ToTable("CollectionTag");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Device", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("EmailAddress")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("IpAddress")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastUsed")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Platform")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId");
 | |
| 
 | |
|                     b.ToTable("Device");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.FolderPath", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("LastScanned")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("LibraryId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Path")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("LibraryId");
 | |
| 
 | |
|                     b.ToTable("FolderPath");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Genre", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("ExternalTag")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("NormalizedTitle")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Title")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("NormalizedTitle", "ExternalTag")
 | |
|                         .IsUnique();
 | |
| 
 | |
|                     b.ToTable("Genre");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Library", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("CoverImage")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastScanned")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Type")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.ToTable("Library");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.MangaFile", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ChapterId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("FilePath")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Format")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("LastFileAnalysis")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Pages")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("ChapterId");
 | |
| 
 | |
|                     b.ToTable("MangaFile");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Metadata.SeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AgeRating")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("AgeRatingLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("CharacterLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("ColoristLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("CoverArtistLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("EditorLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("GenresLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("InkerLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Language")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("LanguageLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("LettererLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("MaxCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("PencillerLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("PublicationStatus")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("PublicationStatusLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("PublisherLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ReleaseYear")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("ReleaseYearLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<uint>("RowVersion")
 | |
|                         .IsConcurrencyToken()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Summary")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("SummaryLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("TagsLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("TotalCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("TranslatorLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("WriterLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("SeriesId")
 | |
|                         .IsUnique();
 | |
| 
 | |
|                     b.HasIndex("Id", "SeriesId")
 | |
|                         .IsUnique();
 | |
| 
 | |
|                     b.ToTable("SeriesMetadata");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Metadata.SeriesRelation", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("RelationKind")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("TargetSeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("SeriesId");
 | |
| 
 | |
|                     b.HasIndex("TargetSeriesId");
 | |
| 
 | |
|                     b.ToTable("SeriesRelation");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Person", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Role")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.ToTable("Person");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.ReadingList", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AgeRating")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("CoverImage")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("CoverImageLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedTitle")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("Promoted")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Summary")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Title")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId");
 | |
| 
 | |
|                     b.ToTable("ReadingList");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.ReadingListItem", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ChapterId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("Order")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("ReadingListId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("VolumeId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("ChapterId");
 | |
| 
 | |
|                     b.HasIndex("ReadingListId");
 | |
| 
 | |
|                     b.HasIndex("SeriesId");
 | |
| 
 | |
|                     b.HasIndex("VolumeId");
 | |
| 
 | |
|                     b.ToTable("ReadingListItem");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Series", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int?>("AppUserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AvgHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("CoverImage")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("CoverImageLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("FolderPath")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Format")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("LastChapterAdded")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastFolderScanned")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("LibraryId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("LocalizedName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("LocalizedNameLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("MaxHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("MinHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("NameLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("NormalizedLocalizedName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("OriginalName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Pages")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("SortName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("SortNameLocked")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<long>("WordCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("AppUserId");
 | |
| 
 | |
|                     b.HasIndex("LibraryId");
 | |
| 
 | |
|                     b.ToTable("Series");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.ServerSetting", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Key")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<uint>("RowVersion")
 | |
|                         .IsConcurrencyToken()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Value")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Key");
 | |
| 
 | |
|                     b.ToTable("ServerSetting");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.SiteTheme", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("FileName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<bool>("IsDefault")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("NormalizedName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Provider")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.ToTable("SiteTheme");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Tag", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<bool>("ExternalTag")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("NormalizedTitle")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Title")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("NormalizedTitle", "ExternalTag")
 | |
|                         .IsUnique();
 | |
| 
 | |
|                     b.ToTable("Tag");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Volume", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("AvgHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("CoverImage")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("Created")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<DateTime>("LastModified")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("MaxHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("MinHoursToRead")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("Number")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("Pages")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<long>("WordCount")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("SeriesId");
 | |
| 
 | |
|                     b.ToTable("Volume");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("AppUserLibrary", b =>
 | |
|                 {
 | |
|                     b.Property<int>("AppUsersId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("LibrariesId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("AppUsersId", "LibrariesId");
 | |
| 
 | |
|                     b.HasIndex("LibrariesId");
 | |
| 
 | |
|                     b.ToTable("AppUserLibrary");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("ChapterGenre", b =>
 | |
|                 {
 | |
|                     b.Property<int>("ChaptersId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("GenresId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("ChaptersId", "GenresId");
 | |
| 
 | |
|                     b.HasIndex("GenresId");
 | |
| 
 | |
|                     b.ToTable("ChapterGenre");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("ChapterPerson", b =>
 | |
|                 {
 | |
|                     b.Property<int>("ChapterMetadatasId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("PeopleId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("ChapterMetadatasId", "PeopleId");
 | |
| 
 | |
|                     b.HasIndex("PeopleId");
 | |
| 
 | |
|                     b.ToTable("ChapterPerson");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("ChapterTag", b =>
 | |
|                 {
 | |
|                     b.Property<int>("ChaptersId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("TagsId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("ChaptersId", "TagsId");
 | |
| 
 | |
|                     b.HasIndex("TagsId");
 | |
| 
 | |
|                     b.ToTable("ChapterTag");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("CollectionTagSeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.Property<int>("CollectionTagsId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesMetadatasId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("CollectionTagsId", "SeriesMetadatasId");
 | |
| 
 | |
|                     b.HasIndex("SeriesMetadatasId");
 | |
| 
 | |
|                     b.ToTable("CollectionTagSeriesMetadata");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("GenreSeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.Property<int>("GenresId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesMetadatasId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("GenresId", "SeriesMetadatasId");
 | |
| 
 | |
|                     b.HasIndex("SeriesMetadatasId");
 | |
| 
 | |
|                     b.ToTable("GenreSeriesMetadata");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<int>", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("ClaimType")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("ClaimValue")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("RoleId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("RoleId");
 | |
| 
 | |
|                     b.ToTable("AspNetRoleClaims", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<int>", b =>
 | |
|                 {
 | |
|                     b.Property<int>("Id")
 | |
|                         .ValueGeneratedOnAdd()
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("ClaimType")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("ClaimValue")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("UserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("Id");
 | |
| 
 | |
|                     b.HasIndex("UserId");
 | |
| 
 | |
|                     b.ToTable("AspNetUserClaims", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<int>", b =>
 | |
|                 {
 | |
|                     b.Property<string>("LoginProvider")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("ProviderKey")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("ProviderDisplayName")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<int>("UserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("LoginProvider", "ProviderKey");
 | |
| 
 | |
|                     b.HasIndex("UserId");
 | |
| 
 | |
|                     b.ToTable("AspNetUserLogins", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<int>", b =>
 | |
|                 {
 | |
|                     b.Property<int>("UserId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<string>("LoginProvider")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Name")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.Property<string>("Value")
 | |
|                         .HasColumnType("TEXT");
 | |
| 
 | |
|                     b.HasKey("UserId", "LoginProvider", "Name");
 | |
| 
 | |
|                     b.ToTable("AspNetUserTokens", (string)null);
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("PersonSeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.Property<int>("PeopleId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("SeriesMetadatasId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("PeopleId", "SeriesMetadatasId");
 | |
| 
 | |
|                     b.HasIndex("SeriesMetadatasId");
 | |
| 
 | |
|                     b.ToTable("PersonSeriesMetadata");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("SeriesMetadataTag", b =>
 | |
|                 {
 | |
|                     b.Property<int>("SeriesMetadatasId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.Property<int>("TagsId")
 | |
|                         .HasColumnType("INTEGER");
 | |
| 
 | |
|                     b.HasKey("SeriesMetadatasId", "TagsId");
 | |
| 
 | |
|                     b.HasIndex("TagsId");
 | |
| 
 | |
|                     b.ToTable("SeriesMetadataTag");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserBookmark", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", "AppUser")
 | |
|                         .WithMany("Bookmarks")
 | |
|                         .HasForeignKey("AppUserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("AppUser");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserPreferences", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", "AppUser")
 | |
|                         .WithOne("UserPreferences")
 | |
|                         .HasForeignKey("API.Entities.AppUserPreferences", "AppUserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.SiteTheme", "Theme")
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("ThemeId");
 | |
| 
 | |
|                     b.Navigation("AppUser");
 | |
| 
 | |
|                     b.Navigation("Theme");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserProgress", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", "AppUser")
 | |
|                         .WithMany("Progresses")
 | |
|                         .HasForeignKey("AppUserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Series", null)
 | |
|                         .WithMany("Progress")
 | |
|                         .HasForeignKey("SeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("AppUser");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserRating", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", "AppUser")
 | |
|                         .WithMany("Ratings")
 | |
|                         .HasForeignKey("AppUserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Series", null)
 | |
|                         .WithMany("Ratings")
 | |
|                         .HasForeignKey("SeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("AppUser");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUserRole", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppRole", "Role")
 | |
|                         .WithMany("UserRoles")
 | |
|                         .HasForeignKey("RoleId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.AppUser", "User")
 | |
|                         .WithMany("UserRoles")
 | |
|                         .HasForeignKey("UserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Role");
 | |
| 
 | |
|                     b.Navigation("User");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Chapter", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Volume", "Volume")
 | |
|                         .WithMany("Chapters")
 | |
|                         .HasForeignKey("VolumeId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Volume");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Device", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", "AppUser")
 | |
|                         .WithMany("Devices")
 | |
|                         .HasForeignKey("AppUserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("AppUser");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.FolderPath", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Library", "Library")
 | |
|                         .WithMany("Folders")
 | |
|                         .HasForeignKey("LibraryId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Library");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.MangaFile", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Chapter", "Chapter")
 | |
|                         .WithMany("Files")
 | |
|                         .HasForeignKey("ChapterId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Chapter");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Metadata.SeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Series", "Series")
 | |
|                         .WithOne("Metadata")
 | |
|                         .HasForeignKey("API.Entities.Metadata.SeriesMetadata", "SeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Series");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Metadata.SeriesRelation", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Series", "Series")
 | |
|                         .WithMany("Relations")
 | |
|                         .HasForeignKey("SeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Series", "TargetSeries")
 | |
|                         .WithMany("RelationOf")
 | |
|                         .HasForeignKey("TargetSeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Series");
 | |
| 
 | |
|                     b.Navigation("TargetSeries");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.ReadingList", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", "AppUser")
 | |
|                         .WithMany("ReadingLists")
 | |
|                         .HasForeignKey("AppUserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("AppUser");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.ReadingListItem", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Chapter", "Chapter")
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("ChapterId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.ReadingList", "ReadingList")
 | |
|                         .WithMany("Items")
 | |
|                         .HasForeignKey("ReadingListId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Series", "Series")
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("SeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Volume", "Volume")
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("VolumeId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Chapter");
 | |
| 
 | |
|                     b.Navigation("ReadingList");
 | |
| 
 | |
|                     b.Navigation("Series");
 | |
| 
 | |
|                     b.Navigation("Volume");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Series", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", null)
 | |
|                         .WithMany("WantToRead")
 | |
|                         .HasForeignKey("AppUserId");
 | |
| 
 | |
|                     b.HasOne("API.Entities.Library", "Library")
 | |
|                         .WithMany("Series")
 | |
|                         .HasForeignKey("LibraryId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Library");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Volume", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Series", "Series")
 | |
|                         .WithMany("Volumes")
 | |
|                         .HasForeignKey("SeriesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.Navigation("Series");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("AppUserLibrary", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("AppUsersId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Library", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("LibrariesId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("ChapterGenre", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Chapter", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("ChaptersId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Genre", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("GenresId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("ChapterPerson", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Chapter", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("ChapterMetadatasId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Person", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("PeopleId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("ChapterTag", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Chapter", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("ChaptersId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Tag", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("TagsId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("CollectionTagSeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.CollectionTag", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("CollectionTagsId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Metadata.SeriesMetadata", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("SeriesMetadatasId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("GenreSeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Genre", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("GenresId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Metadata.SeriesMetadata", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("SeriesMetadatasId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<int>", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppRole", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("RoleId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<int>", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("UserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<int>", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("UserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<int>", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.AppUser", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("UserId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("PersonSeriesMetadata", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Person", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("PeopleId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Metadata.SeriesMetadata", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("SeriesMetadatasId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("SeriesMetadataTag", b =>
 | |
|                 {
 | |
|                     b.HasOne("API.Entities.Metadata.SeriesMetadata", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("SeriesMetadatasId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
| 
 | |
|                     b.HasOne("API.Entities.Tag", null)
 | |
|                         .WithMany()
 | |
|                         .HasForeignKey("TagsId")
 | |
|                         .OnDelete(DeleteBehavior.Cascade)
 | |
|                         .IsRequired();
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppRole", b =>
 | |
|                 {
 | |
|                     b.Navigation("UserRoles");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.AppUser", b =>
 | |
|                 {
 | |
|                     b.Navigation("Bookmarks");
 | |
| 
 | |
|                     b.Navigation("Devices");
 | |
| 
 | |
|                     b.Navigation("Progresses");
 | |
| 
 | |
|                     b.Navigation("Ratings");
 | |
| 
 | |
|                     b.Navigation("ReadingLists");
 | |
| 
 | |
|                     b.Navigation("UserPreferences");
 | |
| 
 | |
|                     b.Navigation("UserRoles");
 | |
| 
 | |
|                     b.Navigation("WantToRead");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Chapter", b =>
 | |
|                 {
 | |
|                     b.Navigation("Files");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Library", b =>
 | |
|                 {
 | |
|                     b.Navigation("Folders");
 | |
| 
 | |
|                     b.Navigation("Series");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.ReadingList", b =>
 | |
|                 {
 | |
|                     b.Navigation("Items");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Series", b =>
 | |
|                 {
 | |
|                     b.Navigation("Metadata");
 | |
| 
 | |
|                     b.Navigation("Progress");
 | |
| 
 | |
|                     b.Navigation("Ratings");
 | |
| 
 | |
|                     b.Navigation("RelationOf");
 | |
| 
 | |
|                     b.Navigation("Relations");
 | |
| 
 | |
|                     b.Navigation("Volumes");
 | |
|                 });
 | |
| 
 | |
|             modelBuilder.Entity("API.Entities.Volume", b =>
 | |
|                 {
 | |
|                     b.Navigation("Chapters");
 | |
|                 });
 | |
| #pragma warning restore 612, 618
 | |
|         }
 | |
|     }
 | |
| }
 |