mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-07-09 03:04:10 -04:00
Add azw3 output plugin (thin wrapper around the standalone kf8 mobi output code)
This commit is contained in:
parent
92c757f0ee
commit
90daad4321
@ -539,7 +539,8 @@ from calibre.ebooks.conversion.plugins.epub_output import EPUBOutput
|
|||||||
from calibre.ebooks.conversion.plugins.fb2_output import FB2Output
|
from calibre.ebooks.conversion.plugins.fb2_output import FB2Output
|
||||||
from calibre.ebooks.conversion.plugins.lit_output import LITOutput
|
from calibre.ebooks.conversion.plugins.lit_output import LITOutput
|
||||||
from calibre.ebooks.conversion.plugins.lrf_output import LRFOutput
|
from calibre.ebooks.conversion.plugins.lrf_output import LRFOutput
|
||||||
from calibre.ebooks.conversion.plugins.mobi_output import MOBIOutput
|
from calibre.ebooks.conversion.plugins.mobi_output import (MOBIOutput,
|
||||||
|
AZW3Output)
|
||||||
from calibre.ebooks.conversion.plugins.oeb_output import OEBOutput
|
from calibre.ebooks.conversion.plugins.oeb_output import OEBOutput
|
||||||
from calibre.ebooks.conversion.plugins.pdb_output import PDBOutput
|
from calibre.ebooks.conversion.plugins.pdb_output import PDBOutput
|
||||||
from calibre.ebooks.conversion.plugins.pdf_output import PDFOutput
|
from calibre.ebooks.conversion.plugins.pdf_output import PDFOutput
|
||||||
@ -580,7 +581,7 @@ plugins += [
|
|||||||
FB2Output,
|
FB2Output,
|
||||||
LITOutput,
|
LITOutput,
|
||||||
LRFOutput,
|
LRFOutput,
|
||||||
MOBIOutput,
|
MOBIOutput, AZW3Output,
|
||||||
OEBOutput,
|
OEBOutput,
|
||||||
PDBOutput,
|
PDBOutput,
|
||||||
PDFOutput,
|
PDFOutput,
|
||||||
|
@ -6,8 +6,32 @@ __license__ = 'GPL v3'
|
|||||||
__copyright__ = '2009, Kovid Goyal <kovid@kovidgoyal.net>'
|
__copyright__ = '2009, Kovid Goyal <kovid@kovidgoyal.net>'
|
||||||
__docformat__ = 'restructuredtext en'
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
from calibre.customize.conversion import OutputFormatPlugin
|
from calibre.customize.conversion import (OutputFormatPlugin,
|
||||||
from calibre.customize.conversion import OptionRecommendation
|
OptionRecommendation)
|
||||||
|
|
||||||
|
def remove_html_cover(oeb, log):
|
||||||
|
from calibre.ebooks.oeb.base import OEB_DOCS
|
||||||
|
|
||||||
|
if not oeb.metadata.cover \
|
||||||
|
or 'cover' not in oeb.guide:
|
||||||
|
return
|
||||||
|
href = oeb.guide['cover'].href
|
||||||
|
del oeb.guide['cover']
|
||||||
|
item = oeb.manifest.hrefs[href]
|
||||||
|
if item.spine_position is not None:
|
||||||
|
log.warn('Found an HTML cover: ', item.href, 'removing it.',
|
||||||
|
'If you find some content missing from the output MOBI, it '
|
||||||
|
'is because you misidentified the HTML cover in the input '
|
||||||
|
'document')
|
||||||
|
oeb.spine.remove(item)
|
||||||
|
if item.media_type in OEB_DOCS:
|
||||||
|
oeb.manifest.remove(item)
|
||||||
|
|
||||||
|
def extract_mobi(output_path, opts):
|
||||||
|
if opts.extract_to is not None:
|
||||||
|
from calibre.ebooks.mobi.debug.main import inspect_mobi
|
||||||
|
ddir = opts.extract_to
|
||||||
|
inspect_mobi(output_path, ddir=ddir)
|
||||||
|
|
||||||
class MOBIOutput(OutputFormatPlugin):
|
class MOBIOutput(OutputFormatPlugin):
|
||||||
|
|
||||||
@ -140,25 +164,6 @@ class MOBIOutput(OutputFormatPlugin):
|
|||||||
# Fix up the periodical href to point to first section href
|
# Fix up the periodical href to point to first section href
|
||||||
toc.nodes[0].href = toc.nodes[0].nodes[0].href
|
toc.nodes[0].href = toc.nodes[0].nodes[0].href
|
||||||
|
|
||||||
def remove_html_cover(self):
|
|
||||||
from calibre.ebooks.oeb.base import OEB_DOCS
|
|
||||||
|
|
||||||
oeb = self.oeb
|
|
||||||
if not oeb.metadata.cover \
|
|
||||||
or 'cover' not in oeb.guide:
|
|
||||||
return
|
|
||||||
href = oeb.guide['cover'].href
|
|
||||||
del oeb.guide['cover']
|
|
||||||
item = oeb.manifest.hrefs[href]
|
|
||||||
if item.spine_position is not None:
|
|
||||||
self.log.warn('Found an HTML cover: ', item.href, 'removing it.',
|
|
||||||
'If you find some content missing from the output MOBI, it '
|
|
||||||
'is because you misidentified the HTML cover in the input '
|
|
||||||
'document')
|
|
||||||
oeb.spine.remove(item)
|
|
||||||
if item.media_type in OEB_DOCS:
|
|
||||||
self.oeb.manifest.remove(item)
|
|
||||||
|
|
||||||
def convert(self, oeb, output_path, input_plugin, opts, log):
|
def convert(self, oeb, output_path, input_plugin, opts, log):
|
||||||
from calibre.utils.config import tweaks
|
from calibre.utils.config import tweaks
|
||||||
from calibre.ebooks.mobi.writer2.resources import Resources
|
from calibre.ebooks.mobi.writer2.resources import Resources
|
||||||
@ -169,7 +174,7 @@ class MOBIOutput(OutputFormatPlugin):
|
|||||||
mobi_type = 'old' # Amazon does not support KF8 periodicals
|
mobi_type = 'old' # Amazon does not support KF8 periodicals
|
||||||
create_kf8 = mobi_type in ('new', 'both')
|
create_kf8 = mobi_type in ('new', 'both')
|
||||||
|
|
||||||
self.remove_html_cover()
|
remove_html_cover(self.oeb, self.log)
|
||||||
resources = Resources(oeb, opts, self.is_periodical,
|
resources = Resources(oeb, opts, self.is_periodical,
|
||||||
add_fonts=create_kf8)
|
add_fonts=create_kf8)
|
||||||
self.check_for_periodical()
|
self.check_for_periodical()
|
||||||
@ -185,7 +190,7 @@ class MOBIOutput(OutputFormatPlugin):
|
|||||||
) if create_kf8 else None
|
) if create_kf8 else None
|
||||||
if mobi_type == 'new':
|
if mobi_type == 'new':
|
||||||
kf8.write(output_path)
|
kf8.write(output_path)
|
||||||
self.extract_mobi(output_path, opts)
|
extract_mobi(output_path, opts)
|
||||||
return
|
return
|
||||||
|
|
||||||
self.log('Creating MOBI 6 output')
|
self.log('Creating MOBI 6 output')
|
||||||
@ -225,11 +230,69 @@ class MOBIOutput(OutputFormatPlugin):
|
|||||||
writer = MobiWriter(opts, resources, kf8,
|
writer = MobiWriter(opts, resources, kf8,
|
||||||
write_page_breaks_after_item=write_page_breaks_after_item)
|
write_page_breaks_after_item=write_page_breaks_after_item)
|
||||||
writer(oeb, output_path)
|
writer(oeb, output_path)
|
||||||
self.extract_mobi(output_path, opts)
|
extract_mobi(output_path, opts)
|
||||||
|
|
||||||
|
class AZW3Output(OutputFormatPlugin):
|
||||||
|
|
||||||
|
name = 'AZW3 Output'
|
||||||
|
author = 'Kovid Goyal'
|
||||||
|
file_type = 'azw3'
|
||||||
|
|
||||||
|
options = set([
|
||||||
|
OptionRecommendation(name='prefer_author_sort',
|
||||||
|
recommended_value=False, level=OptionRecommendation.LOW,
|
||||||
|
help=_('When present, use author sort field as author.')
|
||||||
|
),
|
||||||
|
OptionRecommendation(name='no_inline_toc',
|
||||||
|
recommended_value=False, level=OptionRecommendation.LOW,
|
||||||
|
help=_('Don\'t add Table of Contents to the book. Useful if '
|
||||||
|
'the book has its own table of contents.')),
|
||||||
|
OptionRecommendation(name='toc_title', recommended_value=None,
|
||||||
|
help=_('Title for any generated in-line table of contents.')
|
||||||
|
),
|
||||||
|
OptionRecommendation(name='dont_compress',
|
||||||
|
recommended_value=False, level=OptionRecommendation.LOW,
|
||||||
|
help=_('Disable compression of the file contents.')
|
||||||
|
),
|
||||||
|
OptionRecommendation(name='personal_doc', recommended_value='[PDOC]',
|
||||||
|
help=_('Tag marking book to be filed with Personal Docs')
|
||||||
|
),
|
||||||
|
OptionRecommendation(name='mobi_toc_at_start',
|
||||||
|
recommended_value=False,
|
||||||
|
help=_('When adding the Table of Contents to the book, add it at the start of the '
|
||||||
|
'book instead of the end. Not recommended.')
|
||||||
|
),
|
||||||
|
OptionRecommendation(name='extract_to', recommended_value=None,
|
||||||
|
help=_('Extract the contents of the MOBI file to the'
|
||||||
|
' specified directory. If the directory already '
|
||||||
|
'exists, it will be deleted.')
|
||||||
|
),
|
||||||
|
OptionRecommendation(name='share_not_sync', recommended_value=False,
|
||||||
|
help=_('Enable sharing of book content via Facebook etc. '
|
||||||
|
' on the Kindle. WARNING: Using this feature means that '
|
||||||
|
' the book will not auto sync its last read position '
|
||||||
|
' on multiple devices. Complain to Amazon.')
|
||||||
|
),
|
||||||
|
])
|
||||||
|
|
||||||
|
def convert(self, oeb, output_path, input_plugin, opts, log):
|
||||||
|
from calibre.ebooks.mobi.writer2.resources import Resources
|
||||||
|
from calibre.ebooks.mobi.writer8.main import create_kf8_book
|
||||||
|
|
||||||
|
self.oeb, self.opts, self.log = oeb, opts, log
|
||||||
|
|
||||||
|
resources = Resources(self.oeb, self.opts, self.is_periodical,
|
||||||
|
add_fonts=True, process_images=False)
|
||||||
|
remove_html_cover(self.oeb, self.log)
|
||||||
|
|
||||||
|
# Split on pagebreaks so that the resulting KF8 works better with
|
||||||
|
# calibre's viewer, which does not support CSS page breaks
|
||||||
|
from calibre.ebooks.oeb.transforms.split import Split
|
||||||
|
Split()(self.oeb, self.opts)
|
||||||
|
|
||||||
|
kf8 = create_kf8_book(self.oeb, self.opts, resources, for_joint=False)
|
||||||
|
|
||||||
|
kf8.write(output_path)
|
||||||
|
extract_mobi(output_path, opts)
|
||||||
|
|
||||||
def extract_mobi(self, output_path, opts):
|
|
||||||
if opts.extract_to is not None:
|
|
||||||
from calibre.ebooks.mobi.debug.main import inspect_mobi
|
|
||||||
ddir = opts.extract_to
|
|
||||||
inspect_mobi(output_path, ddir=ddir)
|
|
||||||
|
|
||||||
|
@ -19,9 +19,11 @@ PLACEHOLDER_GIF = b'GIF89a\x01\x00\x01\x00\x80\x00\x00\x00\x00\x00\xff\xff\xff!\
|
|||||||
|
|
||||||
class Resources(object):
|
class Resources(object):
|
||||||
|
|
||||||
def __init__(self, oeb, opts, is_periodical, add_fonts=False):
|
def __init__(self, oeb, opts, is_periodical, add_fonts=False,
|
||||||
|
process_images=True):
|
||||||
self.oeb, self.log, self.opts = oeb, oeb.log, opts
|
self.oeb, self.log, self.opts = oeb, oeb.log, opts
|
||||||
self.is_periodical = is_periodical
|
self.is_periodical = is_periodical
|
||||||
|
self.process_images = True
|
||||||
|
|
||||||
self.item_map = {}
|
self.item_map = {}
|
||||||
self.records = []
|
self.records = []
|
||||||
@ -34,6 +36,8 @@ class Resources(object):
|
|||||||
self.add_resources(add_fonts)
|
self.add_resources(add_fonts)
|
||||||
|
|
||||||
def process_image(self, data):
|
def process_image(self, data):
|
||||||
|
if not self.process_images:
|
||||||
|
return data
|
||||||
return (mobify_image(data) if self.opts.mobi_keep_original_images else
|
return (mobify_image(data) if self.opts.mobi_keep_original_images else
|
||||||
rescale_image(data))
|
rescale_image(data))
|
||||||
|
|
||||||
|
33
src/calibre/gui2/convert/azw3_output.py
Normal file
33
src/calibre/gui2/convert/azw3_output.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# vim:fileencoding=UTF-8:ts=4:sw=4:sta:et:sts=4:ai
|
||||||
|
from __future__ import with_statement
|
||||||
|
|
||||||
|
__license__ = 'GPL v3'
|
||||||
|
__copyright__ = '2009, Kovid Goyal <kovid@kovidgoyal.net>'
|
||||||
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
|
|
||||||
|
from calibre.gui2.convert.azw3_output_ui import Ui_Form
|
||||||
|
from calibre.gui2.convert import Widget
|
||||||
|
|
||||||
|
font_family_model = None
|
||||||
|
|
||||||
|
class PluginWidget(Widget, Ui_Form):
|
||||||
|
|
||||||
|
TITLE = _('AZW3 Output')
|
||||||
|
HELP = _('Options specific to')+' AZW3 '+_('output')
|
||||||
|
COMMIT_NAME = 'azw3_output'
|
||||||
|
ICON = I('mimetypes/mobi.png')
|
||||||
|
|
||||||
|
def __init__(self, parent, get_option, get_help, db=None, book_id=None):
|
||||||
|
Widget.__init__(self, parent,
|
||||||
|
['prefer_author_sort', 'toc_title',
|
||||||
|
'mobi_ignore_margins', 'mobi_toc_at_start',
|
||||||
|
'dont_compress', 'no_inline_toc', 'share_not_sync',
|
||||||
|
'personal_doc']#, 'mobi_navpoints_only_deepest']
|
||||||
|
)
|
||||||
|
self.db, self.book_id = db, book_id
|
||||||
|
|
||||||
|
self.initialize_options(get_option, get_help, db, book_id)
|
||||||
|
|
||||||
|
|
125
src/calibre/gui2/convert/azw3_output.ui
Normal file
125
src/calibre/gui2/convert/azw3_output.ui
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ui version="4.0">
|
||||||
|
<class>Form</class>
|
||||||
|
<widget class="QWidget" name="Form">
|
||||||
|
<property name="geometry">
|
||||||
|
<rect>
|
||||||
|
<x>0</x>
|
||||||
|
<y>0</y>
|
||||||
|
<width>588</width>
|
||||||
|
<height>342</height>
|
||||||
|
</rect>
|
||||||
|
</property>
|
||||||
|
<property name="windowTitle">
|
||||||
|
<string>Form</string>
|
||||||
|
</property>
|
||||||
|
<layout class="QGridLayout" name="gridLayout">
|
||||||
|
<item row="4" column="0" colspan="2">
|
||||||
|
<widget class="QCheckBox" name="opt_prefer_author_sort">
|
||||||
|
<property name="text">
|
||||||
|
<string>Use author &sort for author</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="1" column="0">
|
||||||
|
<widget class="QLabel" name="label">
|
||||||
|
<property name="text">
|
||||||
|
<string>&Title for Table of Contents:</string>
|
||||||
|
</property>
|
||||||
|
<property name="buddy">
|
||||||
|
<cstring>opt_toc_title</cstring>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="1" column="1">
|
||||||
|
<widget class="QLineEdit" name="opt_toc_title"/>
|
||||||
|
</item>
|
||||||
|
<item row="5" column="0">
|
||||||
|
<widget class="QCheckBox" name="opt_dont_compress">
|
||||||
|
<property name="text">
|
||||||
|
<string>Disable compression of the file contents</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="0" column="0">
|
||||||
|
<widget class="QCheckBox" name="opt_no_inline_toc">
|
||||||
|
<property name="text">
|
||||||
|
<string>Do not add Table of Contents to book</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="2" column="0" colspan="2">
|
||||||
|
<widget class="QCheckBox" name="opt_mobi_toc_at_start">
|
||||||
|
<property name="text">
|
||||||
|
<string>Put generated Table of Contents at &start of book instead of end</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="3" column="0">
|
||||||
|
<widget class="QCheckBox" name="opt_mobi_ignore_margins">
|
||||||
|
<property name="text">
|
||||||
|
<string>Ignore &margins</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="7" column="0" colspan="2">
|
||||||
|
<widget class="QGroupBox" name="groupBox">
|
||||||
|
<property name="title">
|
||||||
|
<string>Kindle options</string>
|
||||||
|
</property>
|
||||||
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
|
<item>
|
||||||
|
<layout class="QHBoxLayout" name="horizontalLayout">
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="label_3">
|
||||||
|
<property name="text">
|
||||||
|
<string>Personal Doc tag:</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QLineEdit" name="opt_personal_doc"/>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QCheckBox" name="opt_share_not_sync">
|
||||||
|
<property name="text">
|
||||||
|
<string>Enable sharing of book content via Facebook, etc. WARNING: Disables last read syncing</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<spacer name="verticalSpacer">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>20</width>
|
||||||
|
<height>40</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="8" column="0">
|
||||||
|
<spacer name="verticalSpacer_2">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Vertical</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>20</width>
|
||||||
|
<height>40</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
<resources/>
|
||||||
|
<connections/>
|
||||||
|
</ui>
|
Loading…
x
Reference in New Issue
Block a user