mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-11-19 21:13:02 -05:00
470 lines
17 KiB
Plaintext
470 lines
17 KiB
Plaintext
# vim:fileencoding=utf-8
|
|
# License: GPL v3 Copyright: 2016, Kovid Goyal <kovid at kovidgoyal.net>
|
|
from __python__ import bound_methods, hash_literals
|
|
|
|
from elementmaker import E
|
|
from gettext import gettext as _
|
|
|
|
from book_list.globals import get_read_ui
|
|
from book_list.library_data import sync_library_books
|
|
from book_list.router import home
|
|
from book_list.theme import get_color
|
|
from dom import add_extra_css, build_rule, clear, set_css, svgicon, unique_id
|
|
from modals import error_dialog
|
|
from utils import full_screen_element, request_full_screen, safe_set_inner_html, is_ios
|
|
from read_book.goto import create_goto_panel
|
|
from read_book.prefs.font_size import create_font_size_panel
|
|
from read_book.prefs.main import create_prefs_panel
|
|
from read_book.toc import create_toc_panel
|
|
from read_book.word_actions import create_word_actions_panel
|
|
from session import get_device_uuid
|
|
from widgets import create_button, create_spinner
|
|
|
|
|
|
class LoadingMessage: # {{{
|
|
|
|
def __init__(self, msg):
|
|
self.msg = msg or ''
|
|
|
|
def show(self, container):
|
|
self.container_id = container.getAttribute('id')
|
|
container.style.backgroundColor = get_color('window-background')
|
|
container.appendChild(
|
|
E.div(
|
|
style='text-align:center',
|
|
E.div(create_spinner('100px', '100px')),
|
|
E.h2()
|
|
))
|
|
safe_set_inner_html(container.firstChild.lastChild, self.msg)
|
|
set_css(container.firstChild, position='relative', top='50%', transform='translateY(-50%)')
|
|
|
|
def set_msg(self, msg):
|
|
self.msg = msg
|
|
container = document.getElementById(self.container_id)
|
|
safe_set_inner_html(container.firstChild.lastChild, self.msg)
|
|
|
|
def on_container_click(self, evt):
|
|
pass # Dont allow panel to be closed by a click
|
|
# }}}
|
|
|
|
class DeleteBook: # {{{
|
|
|
|
def __init__(self, overlay, question, ok_icon, ok_text, reload_book):
|
|
self.overlay = overlay
|
|
self.question = question or _(
|
|
'Are you sure you want to remove this book from local storage? You will have to re-download it from calibre if you want to read it again.')
|
|
self.ok_icon = ok_icon or 'trash'
|
|
self.ok_text = ok_text or _('Delete book')
|
|
self.reload_book = reload_book
|
|
|
|
def show(self, container):
|
|
self.container_id = container.getAttribute('id')
|
|
set_css(container, display='flex', justify_content='center', flex_direction='column', background_color=get_color('window-background'))
|
|
container.appendChild(
|
|
E.div(style='margin:1ex 1em',
|
|
E.h2(self.question),
|
|
E.div(style='display:flex; justify-content:flex-end',
|
|
create_button(self.ok_text, self.ok_icon, action=self.delete_book, highlight=True),
|
|
E.span('\xa0'),
|
|
create_button(_('Cancel'), action=self.cancel),
|
|
)
|
|
)
|
|
)
|
|
|
|
def show_working(self):
|
|
container = document.getElementById(self.container_id)
|
|
clear(container)
|
|
container.appendChild(
|
|
E.div(
|
|
style='text-align:center',
|
|
E.div(create_spinner('100px', '100px')),
|
|
E.h2()
|
|
))
|
|
safe_set_inner_html(container.lastChild.lastChild, _('Deleting local book copy, please wait...'))
|
|
|
|
def on_container_click(self, evt):
|
|
pass # Dont allow panel to be closed by a click
|
|
|
|
def delete_book(self):
|
|
self.show_working()
|
|
view = self.overlay.view
|
|
view.ui.db.delete_book(view.book, def(book, errmsg):
|
|
self.overlay.hide_current_panel()
|
|
if errmsg:
|
|
view.ui.show_error(_('Failed to delete book'), _('Failed to delete book from local storage, click "Show details" for more information.'), errmsg)
|
|
else:
|
|
if self.reload_book:
|
|
get_read_ui().reload_book()
|
|
else:
|
|
home()
|
|
)
|
|
|
|
def cancel(self):
|
|
self.overlay.hide_current_panel()
|
|
# }}}
|
|
|
|
class SyncBook: # {{{
|
|
|
|
def __init__(self, overlay):
|
|
self.overlay = overlay
|
|
self.canceled = False
|
|
|
|
def show(self, container):
|
|
self.container_id = container.getAttribute('id')
|
|
set_css(container, display='flex', justify_content='center', flex_direction='column', background_color=get_color('window-background'))
|
|
book = self.overlay.view.book
|
|
to_sync = v'[[book.key, new Date(0)]]'
|
|
sync_library_books(book.key[0], to_sync, self.sync_data_received)
|
|
container.appendChild(
|
|
E.div(style='margin:1ex 1em',
|
|
E.h2(_('Syncing to last read position')),
|
|
E.p(_('Downloading last read data from server, please wait...')),
|
|
E.div(style='display:flex; justify-content:flex-end',
|
|
create_button(_('Cancel'), action=self.cancel),
|
|
)
|
|
)
|
|
)
|
|
|
|
def on_container_click(self, evt):
|
|
pass # Dont allow panel to be closed by a click
|
|
|
|
def cancel(self):
|
|
self.canceled = True
|
|
self.overlay.hide_current_panel()
|
|
|
|
def sync_data_received(self, library_id, lrmap, load_type, xhr, ev):
|
|
if self.canceled:
|
|
return
|
|
self.overlay.hide()
|
|
if load_type is not 'load':
|
|
error_dialog(_('Failed to fetch sync data'), _('Failed to download last read data from server, click "Show details" for more information.'), xhr.error_html)
|
|
return
|
|
data = JSON.parse(xhr.responseText)
|
|
book = self.overlay.view.book
|
|
dev = get_device_uuid()
|
|
epoch = 0
|
|
ans = None
|
|
for key in data:
|
|
book_id, fmt = key.partition(':')[::2]
|
|
if book_id is str(book.key[1]) and fmt.upper() is book.key[2].upper():
|
|
last_read_positions = data[key]
|
|
for d in last_read_positions:
|
|
if d.device is not dev and d.epoch > epoch:
|
|
epoch = d.epoch
|
|
ans = d
|
|
if ans is not None:
|
|
cfi = ans.cfi
|
|
if cfi:
|
|
self.overlay.view.goto_bookpos(cfi)
|
|
|
|
|
|
# }}}
|
|
|
|
# MainOverlay {{{
|
|
|
|
MAIN_OVERLAY_TS_CLASS = 'read-book-main-overlay-top-section'
|
|
MAIN_OVERLAY_ACTIONS_CLASS = 'read-book-main-overlay-actions'
|
|
|
|
add_extra_css(def():
|
|
style = ''
|
|
sel = '.' + MAIN_OVERLAY_TS_CLASS + ' > .' + MAIN_OVERLAY_ACTIONS_CLASS + ' '
|
|
style += build_rule(sel, overflow='hidden')
|
|
sel += '> ul '
|
|
style += build_rule(sel, display='flex', flex_wrap='wrap', list_style='none', border_bottom='1px solid currentColor')
|
|
sel += '> li'
|
|
style += build_rule(sel, border_right='1px solid currentColor', padding='0.5em 1ex', display='flex', align_items='center', cursor='pointer')
|
|
style += build_rule(sel + ':last-child', border_right_style='none')
|
|
style += build_rule(sel + ':hover > *:first-child', color=get_color('window-hover-foreground'))
|
|
style += build_rule(sel + ':active > *:first-child', transform='scale(1.8)')
|
|
return style
|
|
)
|
|
|
|
class MainOverlay:
|
|
|
|
def __init__(self, overlay):
|
|
self.overlay = overlay
|
|
self.timer = None
|
|
self.timer_id = unique_id()
|
|
if window.Intl?.DateTimeFormat:
|
|
self.date_formatter = window.Intl.DateTimeFormat(undefined, {'hour':'numeric', 'minute':'numeric'})
|
|
else:
|
|
self.date_formatter = {'format': def(date):
|
|
return '{}:{}'.format(date.getHours(), date.getMinutes())
|
|
}
|
|
|
|
def show(self, container):
|
|
self.container_id = container.getAttribute('id')
|
|
icon_size = '3.5ex'
|
|
|
|
def ac(text, tooltip, action, icon, is_text_button):
|
|
if is_text_button:
|
|
icon = E.span(icon, style='font-size: 175%; font-weight: bold')
|
|
else:
|
|
icon = svgicon(icon, icon_size, icon_size) if icon else ''
|
|
|
|
return E.li(icon, '\xa0', text, onclick=action, title=tooltip)
|
|
|
|
actions_div = E.div( # actions
|
|
E.ul(
|
|
ac(_('Home'), _('Return to list of books'), def(): home();, 'home'),
|
|
ac(_('Back'), None, self.back, 'arrow-left'),
|
|
ac(_('Forward'), None, self.forward, 'arrow-right'),
|
|
),
|
|
|
|
E.ul(
|
|
ac(_('Search'), _('Search for text in this book'), self.overlay.show_search, 'search'),
|
|
ac(_('Go to'), _('Go to a specific location in the book'), self.overlay.show_goto, 'chevron-right'),
|
|
),
|
|
|
|
E.ul(
|
|
ac(_('Sync'), _('Get last read position and annotations from the server'), self.overlay.sync_book, 'cloud-download'),
|
|
ac(_('Delete'), _('Delete this book from the device'), self.overlay.delete_book, 'trash'),
|
|
ac(_('Reload'), _('Reload this book from the server'), self.overlay.reload_book, 'refresh')
|
|
),
|
|
|
|
E.ul(
|
|
ac(_('Table of Contents'), None, self.overlay.show_toc, 'TC', True),
|
|
# ac(_('Bookmarks'), None, None, 'bookmark'),
|
|
),
|
|
|
|
E.ul(
|
|
ac(_('Font size'), _('Change text size'), self.overlay.show_font_size_chooser, 'Aa', True),
|
|
ac(_('Preferences'), _('Configure the book reader'), self.overlay.show_prefs, 'cogs'),
|
|
),
|
|
|
|
class_=MAIN_OVERLAY_ACTIONS_CLASS
|
|
)
|
|
if not full_screen_element() and not is_ios:
|
|
# No fullscreen on iOS, see http://caniuse.com/#search=fullscreen
|
|
actions_div.appendChild(
|
|
E.ul(
|
|
ac(_('Full screen'), _('Enter full screen mode'), def(): request_full_screen(), self.overlay.hide();, 'full-screen'),
|
|
))
|
|
|
|
container.appendChild(set_css(E.div(class_=MAIN_OVERLAY_TS_CLASS, # top section
|
|
onclick=def (evt):evt.stopPropagation();,
|
|
|
|
set_css(E.div( # top row
|
|
E.div(self.overlay.view.book.metadata.title, style='max-width: 90%; text-overflow: ellipsis; font-weight: bold'),
|
|
E.div(self.date_formatter.format(Date()), id=self.timer_id, style='max-width: 9%; text-overflow: ellipsis'),
|
|
), display='flex', justify_content='space-between', align_items='baseline', font_size='smaller', padding='0.5ex 1rem', border_bottom='solid 1px currentColor'
|
|
),
|
|
actions_div,
|
|
), user_select='none', background_color=get_color('window-background')))
|
|
|
|
container.appendChild(
|
|
set_css(E.div( # bottom bar
|
|
svgicon('close', icon_size, icon_size), '\xa0', _('Close'),
|
|
),
|
|
cursor='pointer', position='fixed', width='100vw', bottom='0', display='flex', justify_content='center', align_items='center', padding='0.5ex 1em',
|
|
user_select='none', background_color=get_color('window-background'),
|
|
)
|
|
)
|
|
|
|
|
|
self.on_hide()
|
|
self.timer = setInterval(self.update_time, 1000)
|
|
|
|
def update_time(self):
|
|
tm = document.getElementById(self.timer_id)
|
|
if tm:
|
|
tm.textContent = self.date_formatter.format(Date())
|
|
|
|
def on_hide(self):
|
|
if self.timer is not None:
|
|
clearInterval(self.timer)
|
|
self.timer = None
|
|
|
|
def back(self):
|
|
window.history.back()
|
|
|
|
def forward(self):
|
|
window.history.forward()
|
|
|
|
# }}}
|
|
|
|
class TOCOverlay: # {{{
|
|
|
|
def __init__(self, overlay, create_func, title):
|
|
self.overlay = overlay
|
|
self.create_func = create_func or create_toc_panel
|
|
self.title = title or _('Table of Contents')
|
|
|
|
def on_container_click(self, evt):
|
|
pass # Dont allow panel to be closed by a click
|
|
|
|
def show(self, container):
|
|
container.style.backgroundColor = get_color('window-background')
|
|
container.appendChild(E.div(
|
|
style='padding: 1ex 1em; border-bottom: solid 1px currentColor; display:flex; justify-content: space-between',
|
|
E.h2(self.title),
|
|
E.div(svgicon('close'), style='cursor:pointer', onclick=def(event):event.preventDefault(), event.stopPropagation(), self.overlay.hide_current_panel(event);, class_='simple-link'),
|
|
))
|
|
self.create_func(self.overlay.view.book, container, self.handle_activate)
|
|
|
|
def handle_activate(self, dest, frag):
|
|
self.overlay.hide()
|
|
if jstype(dest) is 'function':
|
|
dest(self.overlay.view, frag)
|
|
else:
|
|
self.overlay.view.goto_named_destination(dest, frag)
|
|
# }}}
|
|
|
|
class WordActionsOverlay: # {{{
|
|
|
|
def __init__(self, word, overlay):
|
|
self.word = word
|
|
self.overlay = overlay
|
|
|
|
def on_container_click(self, evt):
|
|
pass # Dont allow panel to be closed by a click
|
|
|
|
def show(self, container):
|
|
container.style.backgroundColor = get_color('window-background')
|
|
container.appendChild(E.div(
|
|
style='padding: 1ex 1em; border-bottom: solid 1px currentColor; display:flex; justify-content: space-between',
|
|
E.h2(_('Lookup: {}').format(self.word)),
|
|
E.div(svgicon('close'), style='cursor:pointer', onclick=def(event):event.preventDefault(), event.stopPropagation(), self.overlay.hide_current_panel(event);, class_='simple-link'),
|
|
))
|
|
container.appendChild(E.div())
|
|
create_word_actions_panel(container, self.word, self.overlay.hide)
|
|
# }}}
|
|
|
|
class PrefsOverlay: # {{{
|
|
|
|
def __init__(self, overlay):
|
|
self.overlay = overlay
|
|
self.changes_occurred = False
|
|
|
|
def on_container_click(self, evt):
|
|
pass # Dont allow panel to be closed by a click
|
|
|
|
def show(self, container):
|
|
self.changes_occurred = False
|
|
container.style.backgroundColor = get_color('window-background')
|
|
create_prefs_panel(container, self.overlay.hide_current_panel, def():self.changes_occurred=True;)
|
|
|
|
def on_hide(self):
|
|
if self.changes_occurred:
|
|
self.changes_occurred = False
|
|
get_read_ui().redisplay_book()
|
|
|
|
# }}}
|
|
|
|
class FontSizeOverlay: # {{{
|
|
|
|
def __init__(self, overlay):
|
|
self.overlay = overlay
|
|
|
|
def show(self, container):
|
|
create_font_size_panel(container, self.overlay.hide_current_panel)
|
|
# }}}
|
|
|
|
|
|
class Overlay:
|
|
|
|
def __init__(self, view):
|
|
self.view = view
|
|
c = self.clear_container()
|
|
c.addEventListener('click', self.container_clicked)
|
|
self.panels = []
|
|
|
|
def clear_container(self):
|
|
c = self.container
|
|
clear(c)
|
|
c.style.backgroundColor = 'transparent'
|
|
c.style.color = get_color('window-foreground')
|
|
c.style.display = 'block'
|
|
return c
|
|
|
|
@property
|
|
def container(self):
|
|
return document.getElementById('book-overlay')
|
|
|
|
@property
|
|
def is_visible(self):
|
|
return self.container.style.display is not 'none'
|
|
|
|
def update_visibility(self):
|
|
self.container.style.display = 'block' if self.panels.length else 'none'
|
|
|
|
def container_clicked(self, evt):
|
|
if self.panels.length and jstype(self.panels[-1].on_container_click) is 'function':
|
|
self.panels[-1].on_container_click(evt)
|
|
else:
|
|
self.hide_current_panel()
|
|
|
|
def show_loading_message(self, msg):
|
|
lm = LoadingMessage(msg)
|
|
self.panels.push(lm)
|
|
self.show_current_panel()
|
|
|
|
def hide_loading_message(self):
|
|
self.panels = [p for p in self.panels if not isinstance(p, LoadingMessage)]
|
|
self.show_current_panel()
|
|
|
|
def hide_current_panel(self):
|
|
p = self.panels.pop()
|
|
if p and callable(p.on_hide):
|
|
p.on_hide()
|
|
self.show_current_panel()
|
|
|
|
def show_current_panel(self):
|
|
if self.panels.length:
|
|
c = self.clear_container()
|
|
self.panels[-1].show(c)
|
|
self.update_visibility()
|
|
|
|
def show(self):
|
|
self.panels = [MainOverlay(self)]
|
|
self.show_current_panel()
|
|
|
|
def hide(self):
|
|
while self.panels.length:
|
|
self.hide_current_panel()
|
|
self.update_visibility()
|
|
|
|
def delete_book(self):
|
|
self.hide_current_panel()
|
|
self.panels = [DeleteBook(self)]
|
|
self.show_current_panel()
|
|
|
|
def reload_book(self):
|
|
self.hide_current_panel()
|
|
self.panels = [DeleteBook(self, _('Are you sure you want to reload this book?'), 'refresh', _('Reload book'), True)]
|
|
self.show_current_panel()
|
|
|
|
def sync_book(self):
|
|
self.hide_current_panel()
|
|
self.panels = [SyncBook(self)]
|
|
self.show_current_panel()
|
|
|
|
def show_toc(self):
|
|
self.hide_current_panel()
|
|
self.panels.push(TOCOverlay(self))
|
|
self.show_current_panel()
|
|
|
|
def show_goto(self):
|
|
self.hide_current_panel()
|
|
self.panels.push(TOCOverlay(self, create_goto_panel, _('Go to…')))
|
|
self.show_current_panel()
|
|
|
|
def show_search(self):
|
|
self.hide()
|
|
self.view.show_search()
|
|
|
|
def show_prefs(self):
|
|
self.hide_current_panel()
|
|
self.panels.push(PrefsOverlay(self))
|
|
self.show_current_panel()
|
|
|
|
def show_font_size_chooser(self):
|
|
self.hide_current_panel()
|
|
self.panels.push(FontSizeOverlay(self))
|
|
self.show_current_panel()
|
|
|
|
def show_word_actions(self, word):
|
|
self.hide_current_panel()
|
|
self.panels.push(WordActionsOverlay(word, self))
|
|
self.show_current_panel()
|