mirror of
https://github.com/ajurna/cbwebreader.git
synced 2025-12-06 14:17:19 +00:00
removed os based file access. removed rarfile integration now using package. now using environment variable for comic directory.
381 lines
14 KiB
Python
381 lines
14 KiB
Python
import json
|
|
import uuid
|
|
|
|
from django.contrib.auth import authenticate, login
|
|
from django.contrib.auth.decorators import login_required, user_passes_test
|
|
from django.contrib.auth.models import User
|
|
from django.db.models import Max, Count, F
|
|
from django.db.transaction import atomic
|
|
from django.http import HttpResponse, FileResponse
|
|
from django.shortcuts import get_object_or_404, redirect, render
|
|
from django.utils.http import urlsafe_base64_decode, urlsafe_base64_encode
|
|
from django.views.decorators.csrf import ensure_csrf_cookie
|
|
from django.views.decorators.http import require_POST
|
|
|
|
from .forms import AccountForm, AddUserForm, EditUserForm, InitialSetupForm, SettingsForm
|
|
from .models import ComicBook, ComicPage, ComicStatus, Directory, Setting, UserMisc
|
|
from .util import (
|
|
Menu,
|
|
generate_breadcrumbs_from_menu,
|
|
generate_breadcrumbs_from_path,
|
|
generate_directory,
|
|
generate_label,
|
|
generate_title_from_path,
|
|
)
|
|
|
|
|
|
# noinspection PyTypeChecker
|
|
@ensure_csrf_cookie
|
|
@login_required
|
|
def comic_list(request, directory_selector=False):
|
|
if User.objects.all().count() == 0:
|
|
return redirect("/comic/settings/")
|
|
# try:
|
|
# base_dir = Setting.objects.get(name="BASE_DIR").value
|
|
# except Setting.DoesNotExist:
|
|
# return redirect("/comic/settings/")
|
|
# if not path.isdir(base_dir):
|
|
# return redirect("/comic/settings/")
|
|
|
|
if directory_selector:
|
|
selector = uuid.UUID(bytes=urlsafe_base64_decode(directory_selector))
|
|
directory = Directory.objects.get(selector=selector)
|
|
else:
|
|
directory = False
|
|
|
|
if directory:
|
|
title = generate_title_from_path(directory.path)
|
|
breadcrumbs = generate_breadcrumbs_from_path(directory)
|
|
json_url = "/comic/list_json/{0}/".format(directory_selector)
|
|
else:
|
|
title = generate_title_from_path("Home")
|
|
breadcrumbs = generate_breadcrumbs_from_path()
|
|
json_url = "/comic/list_json/"
|
|
|
|
return render(
|
|
request,
|
|
"comic/comic_list.html",
|
|
{"breadcrumbs": breadcrumbs, "menu": Menu(request.user, "Browse"), "title": title, "json_url": json_url},
|
|
)
|
|
|
|
|
|
@login_required
|
|
@require_POST
|
|
def comic_list_json(request, directory_selector=False):
|
|
icon_str = '<span class="fa {0}"></span>'
|
|
if directory_selector:
|
|
directory_selector = uuid.UUID(bytes=urlsafe_base64_decode(directory_selector))
|
|
directory = Directory.objects.get(selector=directory_selector)
|
|
else:
|
|
directory = False
|
|
files = generate_directory(request.user, directory)
|
|
response_data = dict()
|
|
response_data["data"] = []
|
|
for file in files:
|
|
response_data["data"].append(
|
|
{
|
|
"blank": "",
|
|
"selector": file.selector,
|
|
"type": file.type,
|
|
"icon": icon_str.format(file.icon),
|
|
"name": file.name,
|
|
"label": file.label,
|
|
"url": file.location,
|
|
}
|
|
)
|
|
return HttpResponse(json.dumps(response_data), content_type="application/json")
|
|
|
|
|
|
@login_required
|
|
def recent_comics(request):
|
|
feed_id, _ = UserMisc.objects.get_or_create(user=request.user)
|
|
|
|
return render(
|
|
request,
|
|
"comic/recent_comics.html",
|
|
{
|
|
"breadcrumbs": generate_breadcrumbs_from_menu([("Recent", "/comic/recent/")]),
|
|
"menu": Menu(request.user, "Recent"),
|
|
"title": "Recent Comics",
|
|
"feed_id": urlsafe_base64_encode(feed_id.feed_id.bytes),
|
|
},
|
|
)
|
|
|
|
|
|
@login_required
|
|
@require_POST
|
|
def recent_comics_json(request):
|
|
start = int(request.POST["start"])
|
|
end = start + int(request.POST["length"])
|
|
icon = '<span class="fa fa-book"></span>'
|
|
comics = ComicBook.objects.all().annotate(total_pages=Count('comicpage'))
|
|
response_data = dict()
|
|
response_data["recordsTotal"] = comics.count()
|
|
if request.POST["search[value]"]:
|
|
comics = comics.filter(file_name__contains=request.POST["search[value]"])
|
|
order_string = ""
|
|
# Ordering
|
|
if request.POST["order[0][dir]"] == "desc":
|
|
order_string += "-"
|
|
if request.POST["order[0][dir]"] == "3":
|
|
order_string += "date_added"
|
|
elif request.POST["order[0][dir]"] == "2":
|
|
order_string += "date_added"
|
|
else:
|
|
order_string += "date_added"
|
|
comics = comics.order_by(order_string)
|
|
comics = comics.filter(comicstatus__user=request.user).annotate(
|
|
unread=F('comicstatus__unread'),
|
|
finished=F('comicstatus__finished'),
|
|
last_read_page=F('comicstatus__last_read_page')
|
|
)
|
|
response_data["recordsFiltered"] = comics.count()
|
|
response_data["data"] = list()
|
|
for book in comics[start:end]:
|
|
response_data["data"].append(
|
|
{
|
|
"selector": urlsafe_base64_encode(book.selector.bytes),
|
|
"icon": icon,
|
|
"type": "book",
|
|
"name": book.file_name,
|
|
"date": book.date_added.strftime("%d/%m/%y-%H:%M"),
|
|
"label": generate_label(book),
|
|
"url": "/comic/read/{0}/".format(urlsafe_base64_encode(book.selector.bytes)),
|
|
}
|
|
)
|
|
return HttpResponse(json.dumps(response_data), content_type="application/json")
|
|
|
|
|
|
@login_required
|
|
@require_POST
|
|
def comic_edit(request):
|
|
if "selected" not in request.POST:
|
|
return HttpResponse(status=200)
|
|
if request.POST["func"] == "choose":
|
|
return HttpResponse(status=200)
|
|
selected = [uuid.UUID(bytes=urlsafe_base64_decode(item)) for item in request.POST.getlist("selected")]
|
|
comics = ComicBook.objects.filter(selector__in=selected)
|
|
with atomic():
|
|
for comic in comics:
|
|
status, _ = ComicStatus.objects.get_or_create(comic=comic, user=request.user)
|
|
if request.POST["func"] == "read":
|
|
status.unread = False
|
|
status.finished = True
|
|
status.last_read_page = comic.page_count - 1
|
|
elif request.POST["func"] == "unread":
|
|
status.unread = True
|
|
status.finished = False
|
|
status.last_read_page = 0
|
|
status.save()
|
|
return HttpResponse(status=200)
|
|
|
|
|
|
@login_required
|
|
def account_page(request):
|
|
success_message = []
|
|
if request.POST:
|
|
form = AccountForm(request.POST)
|
|
if form.is_valid():
|
|
if form.cleaned_data["email"] != request.user.email:
|
|
request.user.email = form.cleaned_data["email"]
|
|
success_message.append("Email Updated.")
|
|
if len(form.cleaned_data["password"]) != 0:
|
|
request.user.set_password(form.cleaned_data["password"])
|
|
success_message.append("Password Updated.")
|
|
request.user.save()
|
|
else:
|
|
form = AccountForm(initial={"username": request.user.username, "email": request.user.email})
|
|
crumbs = [("Account", "/comic/account/")]
|
|
context = {
|
|
"form": form,
|
|
"menu": Menu(request.user, "Account"),
|
|
"error_message": form.errors,
|
|
"success_message": "</br>".join(success_message),
|
|
"breadcrumbs": generate_breadcrumbs_from_menu(crumbs),
|
|
"title": "CBWebReader - Account",
|
|
}
|
|
return render(request, "comic/settings_page.html", context)
|
|
|
|
|
|
@user_passes_test(lambda u: u.is_superuser)
|
|
def users_page(request):
|
|
users = User.objects.all()
|
|
crumbs = [("Users", "/comic/settings/users/")]
|
|
context = {
|
|
"users": users,
|
|
"menu": Menu(request.user, "Users"),
|
|
"breadcrumbs": generate_breadcrumbs_from_menu(crumbs),
|
|
}
|
|
return render(request, "comic/users_page.html", context)
|
|
|
|
|
|
@user_passes_test(lambda u: u.is_superuser)
|
|
def user_config_page(request, user_id):
|
|
user = get_object_or_404(User, id=user_id)
|
|
success_message = []
|
|
if request.POST:
|
|
form = EditUserForm(request.POST)
|
|
if form.is_valid():
|
|
if "password" in form.cleaned_data:
|
|
if len(form.cleaned_data["password"]) != 0:
|
|
user.set_password(form.cleaned_data["password"])
|
|
success_message.append("Password Updated.")
|
|
if form.cleaned_data["email"] != user.email:
|
|
user.email = form.cleaned_data["email"]
|
|
success_message.append("Email Updated.</br>")
|
|
user.save()
|
|
else:
|
|
form = EditUserForm(initial=EditUserForm.get_initial_values(user))
|
|
|
|
users = User.objects.all()
|
|
crumbs = [("Users", "/comic/settings/users/"), (user.username, "/comic/settings/users/" + str(user.id))]
|
|
context = {
|
|
"form": form,
|
|
"users": users,
|
|
"menu": Menu(request.user, "Users"),
|
|
"error_message": form.errors,
|
|
"breadcrumbs": generate_breadcrumbs_from_menu(crumbs),
|
|
"success_message": "</br>".join(success_message),
|
|
"title": "CBWebReader - Edit User - " + user.username,
|
|
}
|
|
return render(request, "comic/settings_page.html", context)
|
|
|
|
|
|
@user_passes_test(lambda u: u.is_superuser)
|
|
def user_add_page(request):
|
|
success_message = ""
|
|
if request.POST:
|
|
form = AddUserForm(request.POST)
|
|
if form.is_valid():
|
|
user = User(username=form.cleaned_data["username"], email=form.cleaned_data["email"])
|
|
user.set_password(form.cleaned_data["password"])
|
|
user.save()
|
|
UserMisc.objects.create(user=user)
|
|
success_message = "User {} created.".format(user.username)
|
|
|
|
else:
|
|
form = AddUserForm()
|
|
crumbs = [("Users", "/comic/settings/users/"), ("Add", "/comic/settings/users/add/")]
|
|
context = {
|
|
"form": form,
|
|
"menu": Menu(request.user, "Users"),
|
|
"breadcrumbs": generate_breadcrumbs_from_menu(crumbs),
|
|
"error_message": form.errors,
|
|
"success_message": success_message,
|
|
"title": "CBWebReader - Add User",
|
|
}
|
|
return render(request, "comic/settings_page.html", context)
|
|
|
|
|
|
@user_passes_test(lambda u: u.is_superuser)
|
|
def settings_page(request):
|
|
success_message = []
|
|
crumbs = [("Settings", "/comic/settings/")]
|
|
if request.POST:
|
|
form = SettingsForm(request.POST)
|
|
if form.is_valid():
|
|
base_dir = Setting.objects.get(name="BASE_DIR")
|
|
base_dir.value = form.cleaned_data["base_dir"]
|
|
base_dir.save()
|
|
success_message.append("Settings updated.")
|
|
form = SettingsForm(initial=SettingsForm.get_initial_values())
|
|
context = {
|
|
"error_message": form.errors,
|
|
"success_message": "</br>".join(success_message),
|
|
"form": form,
|
|
"menu": Menu(request.user, "Settings"),
|
|
"title": "CBWebReader - Settings",
|
|
"breadcrumbs": generate_breadcrumbs_from_menu(crumbs),
|
|
}
|
|
return render(request, "comic/settings_page.html", context)
|
|
|
|
|
|
@login_required
|
|
def read_comic(request, comic_selector):
|
|
selector = uuid.UUID(bytes=urlsafe_base64_decode(comic_selector))
|
|
try:
|
|
book = ComicBook.objects.get(selector=selector)
|
|
except ComicBook.DoesNotExist:
|
|
Directory.objects.get(selector=selector)
|
|
return redirect('comic_list', directory_selector=comic_selector)
|
|
except Directory.DoesNotExist:
|
|
return HttpResponse(status=404)
|
|
book = get_object_or_404(ComicBook, selector=selector)
|
|
pages = ComicPage.objects.filter(Comic=book)
|
|
|
|
status, _ = ComicStatus.objects.get_or_create(comic=book, user=request.user)
|
|
title = "CBWebReader - " + book.file_name
|
|
context = {
|
|
"book": book,
|
|
"pages": pages,
|
|
# "orig_file_name": book.page_name(page),
|
|
"nav": book.nav(request.user),
|
|
"status": status,
|
|
"breadcrumbs": generate_breadcrumbs_from_path(book.directory, book),
|
|
"menu": Menu(request.user),
|
|
"title": title,
|
|
}
|
|
if book.file_name.lower().endswith('pdf'):
|
|
context['status'].last_read_page += 1
|
|
return render(request, "comic/read_comic_pdf.html", context)
|
|
else:
|
|
return render(request, "comic/read_comic.html", context)
|
|
|
|
|
|
@login_required
|
|
def set_read_page(request, comic_selector, page):
|
|
page = int(page)
|
|
selector = uuid.UUID(bytes=urlsafe_base64_decode(comic_selector))
|
|
book = get_object_or_404(ComicBook, selector=selector)
|
|
status, _ = ComicStatus.objects.get_or_create(comic=book, user=request.user)
|
|
status.unread = False
|
|
status.last_read_page = page
|
|
if ComicPage.objects.filter(Comic=book).aggregate(Max("index"))["index__max"] == status.last_read_page:
|
|
status.finished = True
|
|
else:
|
|
status.finished = False
|
|
status.save()
|
|
return HttpResponse(status=200)
|
|
|
|
|
|
@login_required
|
|
def get_image(_, comic_selector, page):
|
|
selector = uuid.UUID(bytes=urlsafe_base64_decode(comic_selector))
|
|
book = ComicBook.objects.get(selector=selector)
|
|
img, content = book.get_image(int(page))
|
|
return FileResponse(img, content_type=content)
|
|
|
|
|
|
@login_required
|
|
def get_pdf(_, comic_selector):
|
|
selector = uuid.UUID(bytes=urlsafe_base64_decode(comic_selector))
|
|
book = ComicBook.objects.get(selector=selector)
|
|
return FileResponse(open(book.get_pdf(), 'rb'), content_type='application/pdf')
|
|
|
|
|
|
def initial_setup(request):
|
|
if User.objects.all().exists():
|
|
return redirect("/comic/")
|
|
if request.POST:
|
|
form = InitialSetupForm(request.POST)
|
|
if form.is_valid():
|
|
user = User(
|
|
username=form.cleaned_data["username"],
|
|
email=form.cleaned_data["email"],
|
|
is_staff=True,
|
|
is_superuser=True,
|
|
)
|
|
user.set_password(form.cleaned_data["password"])
|
|
user.save()
|
|
user = authenticate(username=form.cleaned_data["username"], password=form.cleaned_data["password"])
|
|
login(request, user)
|
|
return redirect("/comic/")
|
|
else:
|
|
form = InitialSetupForm()
|
|
context = {"form": form, "title": "CBWebReader - Setup", "error_message": form.errors}
|
|
return render(request, "comic/settings_page.html", context)
|
|
|
|
|
|
def comic_redirect(_):
|
|
return redirect("/comic/")
|