mirror of
https://github.com/janeczku/calibre-web
synced 2024-12-20 23:20:32 +00:00
184 lines
5.9 KiB
Python
184 lines
5.9 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
# This file is part of the Calibre-Web (https://github.com/janeczku/calibre-web)
|
|
# Copyright (C) 2018-2019 OzzieIsaacs, cervinko, jkrehm, bodybybuddha, ok11,
|
|
# andy29485, idalin, Kyosfonica, wuqi, Kennyl, lemmsh,
|
|
# falgh1, grunjol, csitko, ytils, xybydy, trasba, vrabe,
|
|
# ruben-herold, marblepebble, JackED42, SiphonSquirrel,
|
|
# apetresc, nanu-c, mutschler
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
# custom jinja filters
|
|
|
|
from markupsafe import escape
|
|
import datetime
|
|
import mimetypes
|
|
from uuid import uuid4
|
|
|
|
from flask import Blueprint, request, url_for, g
|
|
from flask_babel import format_date
|
|
from .cw_login import current_user
|
|
|
|
from . import constants, logger
|
|
|
|
jinjia = Blueprint('jinjia', __name__)
|
|
log = logger.create()
|
|
|
|
|
|
# pagination links in jinja
|
|
@jinjia.app_template_filter('url_for_other_page')
|
|
def url_for_other_page(page):
|
|
args = request.view_args.copy()
|
|
args['page'] = page
|
|
for get, val in request.args.items():
|
|
if get == "page":
|
|
continue
|
|
args[get] = val
|
|
return url_for(request.endpoint, **args)
|
|
|
|
|
|
# shortentitles to at longest nchar, shorten longer words if necessary
|
|
@jinjia.app_template_filter('shortentitle')
|
|
def shortentitle_filter(s, nchar=20):
|
|
text = s.split()
|
|
res = "" # result
|
|
suml = 0 # overall length
|
|
for line in text:
|
|
if suml >= 60:
|
|
res += '...'
|
|
break
|
|
# if word longer than 20 chars truncate line and append '...', otherwise add whole word to result
|
|
# string, and summarize total length to stop at chars given by nchar
|
|
if len(line) > nchar:
|
|
res += line[:(nchar-3)] + '[..] '
|
|
suml += nchar+3
|
|
else:
|
|
res += line + ' '
|
|
suml += len(line) + 1
|
|
return res.strip()
|
|
|
|
|
|
@jinjia.app_template_filter('mimetype')
|
|
def mimetype_filter(val):
|
|
return mimetypes.types_map.get('.' + val, 'application/octet-stream')
|
|
|
|
|
|
@jinjia.app_template_filter('formatdate')
|
|
def formatdate_filter(val):
|
|
try:
|
|
return format_date(val, format='medium')
|
|
except AttributeError as e:
|
|
log.error('Babel error: %s, Current user locale: %s, Current User: %s', e,
|
|
current_user.locale,
|
|
current_user.name
|
|
)
|
|
return val
|
|
|
|
|
|
@jinjia.app_template_filter('formatdateinput')
|
|
def format_date_input(val):
|
|
input_date = val.isoformat().split('T', 1)[0] # Hack to support dates <1900
|
|
return '' if input_date == "0101-01-01" else input_date
|
|
|
|
|
|
@jinjia.app_template_filter('strftime')
|
|
def timestamptodate(date, fmt=None):
|
|
date = datetime.datetime.fromtimestamp(
|
|
int(date)/1000
|
|
)
|
|
native = date.replace(tzinfo=None)
|
|
if fmt:
|
|
time_format = fmt
|
|
else:
|
|
time_format = '%d %m %Y - %H:%S'
|
|
return native.strftime(time_format)
|
|
|
|
|
|
@jinjia.app_template_filter('yesno')
|
|
def yesno(value, yes, no):
|
|
return yes if value else no
|
|
|
|
|
|
@jinjia.app_template_filter('formatfloat')
|
|
def formatfloat(value, decimals=1):
|
|
if not value or (isinstance(value, str) and not value.is_numeric()):
|
|
return value
|
|
formated_value = ('{0:.' + str(decimals) + 'f}').format(value)
|
|
if formated_value.endswith('.' + "0" * decimals):
|
|
formated_value = formated_value.rstrip('0').rstrip('.')
|
|
return formated_value
|
|
|
|
|
|
@jinjia.app_template_filter('escapedlink')
|
|
def escapedlink_filter(url, text):
|
|
return "<a href='{}'>{}</a>".format(url, escape(text))
|
|
|
|
|
|
@jinjia.app_template_filter('uuidfilter')
|
|
def uuidfilter(var):
|
|
return uuid4()
|
|
|
|
|
|
@jinjia.app_template_filter('cache_timestamp')
|
|
def cache_timestamp(rolling_period='month'):
|
|
if rolling_period == 'day':
|
|
return str(int(datetime.datetime.today().replace(hour=1, minute=1).timestamp()))
|
|
elif rolling_period == 'year':
|
|
return str(int(datetime.datetime.today().replace(day=1).timestamp()))
|
|
else:
|
|
return str(int(datetime.datetime.today().replace(month=1, day=1).timestamp()))
|
|
|
|
|
|
@jinjia.app_template_filter('last_modified')
|
|
def book_last_modified(book):
|
|
return str(int(book.last_modified.timestamp()))
|
|
|
|
|
|
@jinjia.app_template_filter('get_cover_srcset')
|
|
def get_cover_srcset(book):
|
|
srcset = list()
|
|
resolutions = {
|
|
constants.COVER_THUMBNAIL_SMALL: 'sm',
|
|
constants.COVER_THUMBNAIL_MEDIUM: 'md',
|
|
constants.COVER_THUMBNAIL_LARGE: 'lg'
|
|
}
|
|
for resolution, shortname in resolutions.items():
|
|
url = url_for('web.get_cover', book_id=book.id, resolution=shortname, c=book_last_modified(book))
|
|
srcset.append(f'{url} {resolution}x')
|
|
return ', '.join(srcset)
|
|
|
|
|
|
@jinjia.app_template_filter('get_series_srcset')
|
|
def get_cover_srcset(series):
|
|
srcset = list()
|
|
resolutions = {
|
|
constants.COVER_THUMBNAIL_SMALL: 'sm',
|
|
constants.COVER_THUMBNAIL_MEDIUM: 'md',
|
|
constants.COVER_THUMBNAIL_LARGE: 'lg'
|
|
}
|
|
for resolution, shortname in resolutions.items():
|
|
url = url_for('web.get_series_cover', series_id=series.id, resolution=shortname, c=cache_timestamp())
|
|
srcset.append(f'{url} {resolution}x')
|
|
return ', '.join(srcset)
|
|
|
|
|
|
@jinjia.app_template_filter('music')
|
|
def contains_music(book_formats):
|
|
result = False
|
|
for format in book_formats:
|
|
if format.format.lower() in g.constants.EXTENSIONS_AUDIO:
|
|
result = True
|
|
return result
|