from collections import OrderedDict from os import path, listdir from django.db.transaction import atomic from django.utils.http import urlsafe_base64_encode from .models import ComicBook, Setting, ComicStatus, Directory def generate_title_from_path(file_path): if file_path == '': return 'CBWebReader' return 'CBWebReader - ' + ' - '.join(file_path.split(path.sep)) class Menu: def __init__(self, user, page=''): """ :type page: str """ self.menu_items = OrderedDict() self.menu_items['Browse'] = '/comic/' self.menu_items['Recent'] = '/comic/recent/' self.menu_items['Account'] = '/comic/account/' if user.is_superuser: self.menu_items['Settings'] = '/comic/settings/' self.menu_items['Users'] = '/comic/settings/users/' self.menu_items['Logout'] = '/logout/' self.current_page = page class Breadcrumb: def __init__(self): self.name = 'Home' self.url = '/comic/' def __str__(self): return self.name def __unicode__(self): return self.name def generate_breadcrumbs_from_path(directory=False, book=False): """ :type directory: Directory :type book: ComicBook """ output = [Breadcrumb()] if directory: folders = directory.get_path_objects() else: folders = [] for item in folders[::-1]: bc = Breadcrumb() bc.name = item.name bc.url = b'/comic/' + urlsafe_base64_encode(item.selector.bytes) output.append(bc) if book: bc = Breadcrumb() bc.name = book.file_name bc.url = b'/read/' + urlsafe_base64_encode(book.selector.bytes) output.append(bc) return output def generate_breadcrumbs_from_menu(paths): output = [Breadcrumb()] for item in paths: bc = Breadcrumb() bc.name = item[0] bc.url = item[1] output.append(bc) return output class DirFile: def __init__(self): self.name = '' self.isdir = False self.icon = '' self.iscb = False self.location = '' self.label = '' self.cur_page = 0 self.type = '' self.selector = '' def __str__(self): return self.name def generate_directory(user, directory=False): """ :type user: User :type directory: Directory """ base_dir = Setting.objects.get(name='BASE_DIR').value files = [] if directory: dir_path = directory.path ordered_dir_list = get_ordered_dir_list(path.join(base_dir, directory.path)) else: dir_path = '' ordered_dir_list = get_ordered_dir_list(base_dir) with atomic(): for file_name in ordered_dir_list: df = DirFile() df.name = file_name if path.isdir(path.join(base_dir, dir_path, file_name)): try: if directory: d = Directory.objects.get(name=file_name, parent=directory) else: d = Directory.objects.get(name=file_name, parent__isnull=True) except Directory.DoesNotExist: if directory: d = Directory(name=file_name, parent=directory) else: d = Directory(name=file_name) d.save() df.isdir = True df.icon = 'glyphicon-folder-open' df.selector = urlsafe_base64_encode(d.selector.bytes).decode() df.location = '/comic/{0}/'.format(df.selector) df.label = generate_dir_status(user, d) df.type = 'directory' elif file_name.lower()[-4:] in ['.rar', '.zip', '.cbr', '.cbz']: df.iscb = True df.icon = 'glyphicon-book' try: if directory: book = ComicBook.objects.get(file_name=file_name, directory=directory) else: book = ComicBook.objects.get(file_name=file_name, directory__isnull=True) except ComicBook.DoesNotExist: book = ComicBook.process_comic_book(file_name, directory) status, created = ComicStatus.objects.get_or_create(comic=book, user=user) if created: status.save() last_page = status.last_read_page df.selector = urlsafe_base64_encode(book.selector.bytes).decode() df.location = '/comic/read/{0}/{1}/'.format(df.selector, last_page) df.cur_page = last_page df.label = generate_label(book, status) df.type = 'book' files.append(df) return files def generate_label(book, status): if status.unread: label_text = '
Unread
' elif (status.last_read_page + 1) == book.page_count: label_text = '
Read
' else: label_text = '
%s/%s
' % \ (status.last_read_page + 1, book.page_count) return label_text def generate_dir_status(user, directory): cb_list = ComicBook.objects.filter(directory=directory) total = cb_list.count() total_read = ComicStatus.objects.filter(user=user, comic__in=cb_list, finished=True).count() if total == 0: return '
Empty
' elif total == total_read: return '
All Read
' elif total_read == 0: return '
Unread
' return '
{0}/{1}
'.format(total_read, total) def get_ordered_dir_list(folder): directories = [] files = [] for item in listdir(folder): if path.isdir(path.join(folder, item)): directories.append(item) else: files.append(item) return sorted(directories) + sorted(files)