mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-11-03 19:17:05 -05:00 
			
		
		
		
	* Started designing the backend localization service * Worked in Transloco for initial PoC * Worked in Transloco for initial PoC * Translated the login screen * translated dashboard screen * Started work on the backend * Fixed a logic bug * translated edit-user screen * Hooked up the backend for having a locale property. * Hooked up the ability to view the available locales and switch to them. * Made the localization service languages be derived from what's in langs/ directory. * Fixed up localization switching * Switched when we check for a license on UI bootstrap * Tweaked some code * Fixed the bug where dashboard wasn't loading and made it so language switching is working. * Fixed a bug on dashboard with languagePath * Converted user-scrobble-history.component.html * Converted spoiler.component.html * Converted review-series-modal.component.html * Converted review-card-modal.component.html * Updated the readme * Translated using Weblate (English) Currently translated at 100.0% (54 of 54 strings) Translation: Kavita/ui Translate-URL: https://hosted.weblate.org/projects/kavita/ui/en/ * Converted review-card.component.html * Deleted dead component * Converted want-to-read.component.html * Added translation using Weblate (Korean) * Translated using Weblate (Spanish) Currently translated at 40.7% (22 of 54 strings) Translation: Kavita/ui Translate-URL: https://hosted.weblate.org/projects/kavita/ui/es/ * Translated using Weblate (Korean) Currently translated at 62.9% (34 of 54 strings) Translation: Kavita/ui Translate-URL: https://hosted.weblate.org/projects/kavita/ui/ko/ * Converted user-preferences.component.html * Translated using Weblate (Korean) Currently translated at 92.5% (50 of 54 strings) Translation: Kavita/ui Translate-URL: https://hosted.weblate.org/projects/kavita/ui/ko/ * Converted user-holds.component.html * Converted theme-manager.component.html * Converted restriction-selector.component.html * Converted manage-devices.component.html * Converted edit-device.component.html * Converted change-password.component.html * Converted change-email.component.html * Converted change-age-restriction.component.html * Converted api-key.component.html * Converted anilist-key.component.html * Converted typeahead.component.html * Converted user-stats-info-cards.component.html * Converted user-stats.component.html * Converted top-readers.component.html * Converted some pipes and ensure translation is loaded before the app. * Finished all but one pipe for localization * Converted directory-picker.component.html * Converted library-access-modal.component.html * Converted a few components * Converted a few components * Converted a few components * Converted a few components * Converted a few components * Merged weblate in * ... -> … update * Updated the readme * Updateded all fonts to be woff2 * Cleaned up some strings to increase re-use * Removed an old flow (that doesn't exist in backend any longer) from when we introduced emails on Kavita. * Converted Series detail * Lots more converted * Lots more converted & hooked up the ability to flatten during prod build the language files. * Lots more converted * Lots more converted & fixed a bunch of broken pipes due to inject() * Lots more converted * Lots more converted * Lots more converted & fixed some bad keys * Lots more converted * Fixed some bugs with admin dasbhoard nested tabs not rendering on first load due to not using onpush change detection * Fixed up some localization errors and fixed forgot password error when the user doesn't have change password permission * Fixed a stupid build issue again * Started adding errors for interceptor and backend. * Finished off manga-reader * More translations * Few fixes * Fixed a bug where character tag badges weren't showing the name on chapter info * All components are translated * All toasts are translated * All confirm/alerts are translated * Trying something new for the backend * Migrated the localization strings for the backend into a new file. * Updated the localization service to be able to do backend localization with fallback to english. * Cleaned up some external reviews code to reduce looping * Localized AccountController.cs * 60% done with controllers * All controllers are done * All KavitaExceptions are covered * Some shakeout fixes * Prep for initial merge * Everything is done except options and basic shakeout proves response times are good. Unit tests are broken. * Fixed up the unit tests * All unit tests are now working * Removed some quantifier * I'm not sure I can support localization for some Volume/Chapter/Book strings within the codebase. --------- Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: majora2007 <kavitareader@gmail.com> Co-authored-by: expertjun <jtrobin@naver.com> Co-authored-by: ThePromidius <thepromidiusyt@gmail.com>
		
			
				
	
	
		
			497 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			497 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
using System.Collections.Generic;
 | 
						|
using System.Linq;
 | 
						|
using System.Threading.Tasks;
 | 
						|
using API.Constants;
 | 
						|
using API.Data;
 | 
						|
using API.Data.Repositories;
 | 
						|
using API.DTOs;
 | 
						|
using API.DTOs.ReadingLists;
 | 
						|
using API.Extensions;
 | 
						|
using API.Helpers;
 | 
						|
using API.Services;
 | 
						|
using API.SignalR;
 | 
						|
using Kavita.Common;
 | 
						|
using Microsoft.AspNetCore.Authorization;
 | 
						|
using Microsoft.AspNetCore.Mvc;
 | 
						|
 | 
						|
namespace API.Controllers;
 | 
						|
 | 
						|
[Authorize]
 | 
						|
public class ReadingListController : BaseApiController
 | 
						|
{
 | 
						|
    private readonly IUnitOfWork _unitOfWork;
 | 
						|
    private readonly IReadingListService _readingListService;
 | 
						|
    private readonly ILocalizationService _localizationService;
 | 
						|
 | 
						|
    public ReadingListController(IUnitOfWork unitOfWork, IReadingListService readingListService,
 | 
						|
        ILocalizationService localizationService)
 | 
						|
    {
 | 
						|
        _unitOfWork = unitOfWork;
 | 
						|
        _readingListService = readingListService;
 | 
						|
        _localizationService = localizationService;
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Fetches a single Reading List
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpGet]
 | 
						|
    public async Task<ActionResult<IEnumerable<ReadingListDto>>> GetList(int readingListId)
 | 
						|
    {
 | 
						|
        var userId = await _unitOfWork.UserRepository.GetUserIdByUsernameAsync(User.GetUsername());
 | 
						|
        return Ok(await _unitOfWork.ReadingListRepository.GetReadingListDtoByIdAsync(readingListId, userId));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Returns reading lists (paginated) for a given user.
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="includePromoted">Include Promoted Reading Lists along with user's Reading Lists. Defaults to true</param>
 | 
						|
    /// <param name="userParams">Pagination parameters</param>
 | 
						|
    /// <param name="sortByLastModified">Sort by last modified (most recent first) or by title (alphabetical)</param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("lists")]
 | 
						|
    public async Task<ActionResult<IEnumerable<ReadingListDto>>> GetListsForUser([FromQuery] UserParams userParams,
 | 
						|
        bool includePromoted = true, bool sortByLastModified = false)
 | 
						|
    {
 | 
						|
        var userId = await _unitOfWork.UserRepository.GetUserIdByUsernameAsync(User.GetUsername());
 | 
						|
        var items = await _unitOfWork.ReadingListRepository.GetReadingListDtosForUserAsync(userId, includePromoted,
 | 
						|
            userParams, sortByLastModified);
 | 
						|
        Response.AddPaginationHeader(items.CurrentPage, items.PageSize, items.TotalCount, items.TotalPages);
 | 
						|
 | 
						|
        return Ok(items);
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Returns all Reading Lists the user has access to that have a series within it.
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="seriesId"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpGet("lists-for-series")]
 | 
						|
    public async Task<ActionResult<IEnumerable<ReadingListDto>>> GetListsForSeries(int seriesId)
 | 
						|
    {
 | 
						|
        var userId = await _unitOfWork.UserRepository.GetUserIdByUsernameAsync(User.GetUsername());
 | 
						|
        var items = await _unitOfWork.ReadingListRepository.GetReadingListDtosForSeriesAndUserAsync(userId, seriesId, true);
 | 
						|
 | 
						|
        return Ok(items);
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Fetches all reading list items for a given list including rich metadata around series, volume, chapters, and progress
 | 
						|
    /// </summary>
 | 
						|
    /// <remarks>This call is expensive</remarks>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpGet("items")]
 | 
						|
    public async Task<ActionResult<IEnumerable<ReadingListItemDto>>> GetListForUser(int readingListId)
 | 
						|
    {
 | 
						|
        var items = await _unitOfWork.ReadingListRepository.GetReadingListItemDtosByIdAsync(readingListId, User.GetUserId());
 | 
						|
        return Ok(items);
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Updates an items position
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("update-position")]
 | 
						|
    public async Task<ActionResult> UpdateListItemPosition(UpdateReadingListPosition dto)
 | 
						|
    {
 | 
						|
        // Make sure UI buffers events
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
 | 
						|
        if (await _readingListService.UpdateReadingListItemPosition(dto)) return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
 | 
						|
 | 
						|
        return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-position"));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Deletes a list item from the list. Will reorder all item positions afterwards
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("delete-item")]
 | 
						|
    public async Task<ActionResult> DeleteListItem(UpdateReadingListPosition dto)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
 | 
						|
        if (await _readingListService.DeleteReadingListItem(dto))
 | 
						|
        {
 | 
						|
            return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
        }
 | 
						|
 | 
						|
        return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-item-delete"));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Removes all entries that are fully read from the reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("remove-read")]
 | 
						|
    public async Task<ActionResult> DeleteReadFromList([FromQuery] int readingListId)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(readingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
 | 
						|
        if (await _readingListService.RemoveFullyReadItems(readingListId, user))
 | 
						|
        {
 | 
						|
            return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
        }
 | 
						|
 | 
						|
        return BadRequest("Couldn't delete item(s)");
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Deletes a reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpDelete]
 | 
						|
    public async Task<ActionResult> DeleteList([FromQuery] int readingListId)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(readingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
 | 
						|
        if (await _readingListService.DeleteReadingList(readingListId, user))
 | 
						|
            return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-deleted"));
 | 
						|
 | 
						|
        return BadRequest(await _localizationService.Translate(User.GetUserId(), "generic-reading-list-delete"));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Creates a new List with a unique title. Returns the new ReadingList back
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("create")]
 | 
						|
    public async Task<ActionResult<ReadingListDto>> CreateList(CreateReadingListDto dto)
 | 
						|
    {
 | 
						|
        var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync(User.GetUsername(), AppUserIncludes.ReadingLists);
 | 
						|
        if (user == null) return Unauthorized();
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            await _readingListService.CreateReadingListForUser(user, dto.Title);
 | 
						|
        }
 | 
						|
        catch (KavitaException ex)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), ex.Message));
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _unitOfWork.ReadingListRepository.GetReadingListDtoByTitleAsync(user.Id, dto.Title));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Update the properties (title, summary) of a reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("update")]
 | 
						|
    public async Task<ActionResult> UpdateList(UpdateReadingListDto dto)
 | 
						|
    {
 | 
						|
        var readingList = await _unitOfWork.ReadingListRepository.GetReadingListByIdAsync(dto.ReadingListId);
 | 
						|
        if (readingList == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-doesnt-exist"));
 | 
						|
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(readingList.Id, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            await _readingListService.UpdateReadingList(readingList, dto);
 | 
						|
        }
 | 
						|
        catch (KavitaException ex)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), ex.Message));
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Adds all chapters from a Series to a reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("update-by-series")]
 | 
						|
    public async Task<ActionResult> UpdateListBySeries(UpdateReadingListBySeriesDto dto)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
 | 
						|
        var readingList = user.ReadingLists.SingleOrDefault(l => l.Id == dto.ReadingListId);
 | 
						|
        if (readingList == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-doesnt-exist"));
 | 
						|
        var chapterIdsForSeries =
 | 
						|
            await _unitOfWork.SeriesRepository.GetChapterIdsForSeriesAsync(new [] {dto.SeriesId});
 | 
						|
 | 
						|
        // If there are adds, tell tracking this has been modified
 | 
						|
        if (await _readingListService.AddChaptersToReadingList(dto.SeriesId, chapterIdsForSeries, readingList))
 | 
						|
        {
 | 
						|
            _unitOfWork.ReadingListRepository.Update(readingList);
 | 
						|
        }
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (_unitOfWork.HasChanges())
 | 
						|
            {
 | 
						|
                await _unitOfWork.CommitAsync();
 | 
						|
                return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch
 | 
						|
        {
 | 
						|
            await _unitOfWork.RollbackAsync();
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _localizationService.Translate(User.GetUserId(), "nothing-to-do"));
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Adds all chapters from a list of volumes and chapters to a reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("update-by-multiple")]
 | 
						|
    public async Task<ActionResult> UpdateListByMultiple(UpdateReadingListByMultipleDto dto)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
        var readingList = user.ReadingLists.SingleOrDefault(l => l.Id == dto.ReadingListId);
 | 
						|
        if (readingList == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-doesnt-exist"));
 | 
						|
 | 
						|
        var chapterIds = await _unitOfWork.VolumeRepository.GetChapterIdsByVolumeIds(dto.VolumeIds);
 | 
						|
        foreach (var chapterId in dto.ChapterIds)
 | 
						|
        {
 | 
						|
            chapterIds.Add(chapterId);
 | 
						|
        }
 | 
						|
 | 
						|
        // If there are adds, tell tracking this has been modified
 | 
						|
        if (await _readingListService.AddChaptersToReadingList(dto.SeriesId, chapterIds, readingList))
 | 
						|
        {
 | 
						|
            _unitOfWork.ReadingListRepository.Update(readingList);
 | 
						|
        }
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (_unitOfWork.HasChanges())
 | 
						|
            {
 | 
						|
                await _unitOfWork.CommitAsync();
 | 
						|
                return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch
 | 
						|
        {
 | 
						|
            await _unitOfWork.RollbackAsync();
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _localizationService.Translate(User.GetUserId(), "nothing-to-do"));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Adds all chapters from a list of series to a reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="dto"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpPost("update-by-multiple-series")]
 | 
						|
    public async Task<ActionResult> UpdateListByMultipleSeries(UpdateReadingListByMultipleSeriesDto dto)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
        var readingList = user.ReadingLists.SingleOrDefault(l => l.Id == dto.ReadingListId);
 | 
						|
        if (readingList == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-doesnt-exist"));
 | 
						|
 | 
						|
        var ids = await _unitOfWork.SeriesRepository.GetChapterIdWithSeriesIdForSeriesAsync(dto.SeriesIds.ToArray());
 | 
						|
 | 
						|
        foreach (var seriesId in ids.Keys)
 | 
						|
        {
 | 
						|
            // If there are adds, tell tracking this has been modified
 | 
						|
            if (await _readingListService.AddChaptersToReadingList(seriesId, ids[seriesId], readingList))
 | 
						|
            {
 | 
						|
                _unitOfWork.ReadingListRepository.Update(readingList);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (_unitOfWork.HasChanges())
 | 
						|
            {
 | 
						|
                await _unitOfWork.CommitAsync();
 | 
						|
                return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch
 | 
						|
        {
 | 
						|
            await _unitOfWork.RollbackAsync();
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _localizationService.Translate(User.GetUserId(), "nothing-to-do"));
 | 
						|
    }
 | 
						|
 | 
						|
    [HttpPost("update-by-volume")]
 | 
						|
    public async Task<ActionResult> UpdateListByVolume(UpdateReadingListByVolumeDto dto)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
        var readingList = user.ReadingLists.SingleOrDefault(l => l.Id == dto.ReadingListId);
 | 
						|
        if (readingList == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-doesnt-exist"));
 | 
						|
 | 
						|
        var chapterIdsForVolume =
 | 
						|
            (await _unitOfWork.ChapterRepository.GetChaptersAsync(dto.VolumeId)).Select(c => c.Id).ToList();
 | 
						|
 | 
						|
        // If there are adds, tell tracking this has been modified
 | 
						|
        if (await _readingListService.AddChaptersToReadingList(dto.SeriesId, chapterIdsForVolume, readingList))
 | 
						|
        {
 | 
						|
            _unitOfWork.ReadingListRepository.Update(readingList);
 | 
						|
        }
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (_unitOfWork.HasChanges())
 | 
						|
            {
 | 
						|
                await _unitOfWork.CommitAsync();
 | 
						|
                return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch
 | 
						|
        {
 | 
						|
            await _unitOfWork.RollbackAsync();
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _localizationService.Translate(User.GetUserId(), "nothing-to-do"));
 | 
						|
    }
 | 
						|
 | 
						|
    [HttpPost("update-by-chapter")]
 | 
						|
    public async Task<ActionResult> UpdateListByChapter(UpdateReadingListByChapterDto dto)
 | 
						|
    {
 | 
						|
        var user = await _readingListService.UserHasReadingListAccess(dto.ReadingListId, User.GetUsername());
 | 
						|
        if (user == null)
 | 
						|
        {
 | 
						|
            return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-permission"));
 | 
						|
        }
 | 
						|
        var readingList = user.ReadingLists.SingleOrDefault(l => l.Id == dto.ReadingListId);
 | 
						|
        if (readingList == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "reading-list-doesnt-exist"));
 | 
						|
 | 
						|
        // If there are adds, tell tracking this has been modified
 | 
						|
        if (await _readingListService.AddChaptersToReadingList(dto.SeriesId, new List<int>() { dto.ChapterId }, readingList))
 | 
						|
        {
 | 
						|
            _unitOfWork.ReadingListRepository.Update(readingList);
 | 
						|
        }
 | 
						|
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (_unitOfWork.HasChanges())
 | 
						|
            {
 | 
						|
                await _unitOfWork.CommitAsync();
 | 
						|
                return Ok(await _localizationService.Translate(User.GetUserId(), "reading-list-updated"));
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch
 | 
						|
        {
 | 
						|
            await _unitOfWork.RollbackAsync();
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(await _localizationService.Translate(User.GetUserId(), "nothing-to-do"));
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Returns a list of characters associated with the reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [HttpGet("characters")]
 | 
						|
    [ResponseCache(CacheProfileName = ResponseCacheProfiles.TenMinute)]
 | 
						|
    public ActionResult<IEnumerable<PersonDto>> GetCharactersForList(int readingListId)
 | 
						|
    {
 | 
						|
        return Ok(_unitOfWork.ReadingListRepository.GetReadingListCharactersAsync(readingListId));
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Returns the next chapter within the reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="currentChapterId"></param>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns>Chapter Id for next item, -1 if nothing exists</returns>
 | 
						|
    [HttpGet("next-chapter")]
 | 
						|
    public async Task<ActionResult<int>> GetNextChapter(int currentChapterId, int readingListId)
 | 
						|
    {
 | 
						|
        var items = (await _unitOfWork.ReadingListRepository.GetReadingListItemsByIdAsync(readingListId)).ToList();
 | 
						|
        var readingListItem = items.SingleOrDefault(rl => rl.ChapterId == currentChapterId);
 | 
						|
        if (readingListItem == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "chapter-doesnt-exist"));
 | 
						|
        var index = items.IndexOf(readingListItem) + 1;
 | 
						|
        if (items.Count > index)
 | 
						|
        {
 | 
						|
            return items[index].ChapterId;
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(-1);
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Returns the prev chapter within the reading list
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="currentChapterId"></param>
 | 
						|
    /// <param name="readingListId"></param>
 | 
						|
    /// <returns>Chapter Id for next item, -1 if nothing exists</returns>
 | 
						|
    [HttpGet("prev-chapter")]
 | 
						|
    public async Task<ActionResult<int>> GetPrevChapter(int currentChapterId, int readingListId)
 | 
						|
    {
 | 
						|
        var items = (await _unitOfWork.ReadingListRepository.GetReadingListItemsByIdAsync(readingListId)).ToList();
 | 
						|
        var readingListItem = items.SingleOrDefault(rl => rl.ChapterId == currentChapterId);
 | 
						|
        if (readingListItem == null) return BadRequest(await _localizationService.Translate(User.GetUserId(), "chapter-doesnt-exist"));
 | 
						|
        var index = items.IndexOf(readingListItem) - 1;
 | 
						|
        if (0 <= index)
 | 
						|
        {
 | 
						|
            return items[index].ChapterId;
 | 
						|
        }
 | 
						|
 | 
						|
        return Ok(-1);
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Checks if a reading list exists with the name
 | 
						|
    /// </summary>
 | 
						|
    /// <param name="name">If empty or null, will return true as that is invalid</param>
 | 
						|
    /// <returns></returns>
 | 
						|
    [Authorize(Policy = "RequireAdminRole")]
 | 
						|
    [HttpGet("name-exists")]
 | 
						|
    public async Task<ActionResult<bool>> DoesNameExists(string name)
 | 
						|
    {
 | 
						|
        if (string.IsNullOrEmpty(name)) return true;
 | 
						|
        return Ok(await _unitOfWork.ReadingListRepository.ReadingListExists(name));
 | 
						|
    }
 | 
						|
}
 |