mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-07-09 03:04:10 -04:00
KG updates
This commit is contained in:
commit
22ff1d94d7
@ -1164,6 +1164,12 @@ class StoreFoylesUKStore(StoreBase):
|
|||||||
description = _('Foyles of London, online.')
|
description = _('Foyles of London, online.')
|
||||||
actual_plugin = 'calibre.gui2.store.foyles_uk_plugin:FoylesUKStore'
|
actual_plugin = 'calibre.gui2.store.foyles_uk_plugin:FoylesUKStore'
|
||||||
|
|
||||||
|
class StoreGandalfStore(StoreBase):
|
||||||
|
name = 'Gandalf'
|
||||||
|
author = 'Tomasz Długosz'
|
||||||
|
description = _('Zaczarowany świat książek')
|
||||||
|
actual_plugin = 'calibre.gui2.store.gandalf_plugin:GandalfStore'
|
||||||
|
|
||||||
class StoreGoogleBooksStore(StoreBase):
|
class StoreGoogleBooksStore(StoreBase):
|
||||||
name = 'Google Books'
|
name = 'Google Books'
|
||||||
description = _('Google Books')
|
description = _('Google Books')
|
||||||
@ -1191,6 +1197,7 @@ class StoreMobileReadStore(StoreBase):
|
|||||||
|
|
||||||
class StoreNextoStore(StoreBase):
|
class StoreNextoStore(StoreBase):
|
||||||
name = 'Nexto'
|
name = 'Nexto'
|
||||||
|
author = 'Tomasz Długosz'
|
||||||
description = _('Audiobooki mp3, ebooki, prasa - księgarnia internetowa.')
|
description = _('Audiobooki mp3, ebooki, prasa - księgarnia internetowa.')
|
||||||
actual_plugin = 'calibre.gui2.store.nexto_plugin:NextoStore'
|
actual_plugin = 'calibre.gui2.store.nexto_plugin:NextoStore'
|
||||||
|
|
||||||
@ -1204,6 +1211,11 @@ class StoreOReillyStore(StoreBase):
|
|||||||
description = _('DRM-Free tech ebooks.')
|
description = _('DRM-Free tech ebooks.')
|
||||||
actual_plugin = 'calibre.gui2.store.oreilly_plugin:OReillyStore'
|
actual_plugin = 'calibre.gui2.store.oreilly_plugin:OReillyStore'
|
||||||
|
|
||||||
|
class StorePragmaticBookshelfStore(StoreBase):
|
||||||
|
name = 'Pragmatic Bookshelf'
|
||||||
|
description = _('The Pragmatic Bookshelf')
|
||||||
|
actual_plugin = 'calibre.gui2.store.pragmatic_bookshelf_plugin:PragmaticBookshelfStore'
|
||||||
|
|
||||||
class StoreSmashwordsStore(StoreBase):
|
class StoreSmashwordsStore(StoreBase):
|
||||||
name = 'Smashwords'
|
name = 'Smashwords'
|
||||||
description = _('Your ebook. Your way.')
|
description = _('Your ebook. Your way.')
|
||||||
@ -1224,15 +1236,35 @@ class StoreWizardsTowerBooksStore(StoreBase):
|
|||||||
description = 'Wizard\'s Tower Press.'
|
description = 'Wizard\'s Tower Press.'
|
||||||
actual_plugin = 'calibre.gui2.store.wizards_tower_books_plugin:WizardsTowerBooksStore'
|
actual_plugin = 'calibre.gui2.store.wizards_tower_books_plugin:WizardsTowerBooksStore'
|
||||||
|
|
||||||
plugins += [StoreArchiveOrgStore, StoreAmazonKindleStore, StoreAmazonDEKindleStore,
|
plugins += [
|
||||||
StoreAmazonUKKindleStore, StoreBaenWebScriptionStore, StoreBNStore,
|
StoreArchiveOrgStore,
|
||||||
StoreBeamEBooksDEStore, StoreBeWriteStore,
|
StoreAmazonKindleStore,
|
||||||
StoreDieselEbooksStore, StoreEbookscomStore, StoreEPubBuyDEStore,
|
StoreAmazonDEKindleStore,
|
||||||
StoreEHarlequinStore, StoreFeedbooksStore,
|
StoreAmazonUKKindleStore,
|
||||||
StoreFoylesUKStore, StoreGoogleBooksStore, StoreGutenbergStore,
|
StoreBaenWebScriptionStore,
|
||||||
StoreKoboStore, StoreManyBooksStore,
|
StoreBNStore,
|
||||||
StoreMobileReadStore, StoreNextoStore, StoreOpenLibraryStore,
|
StoreBeamEBooksDEStore,
|
||||||
StoreOReillyStore, StoreSmashwordsStore,
|
StoreBeWriteStore,
|
||||||
StoreWaterstonesUKStore, StoreWeightlessBooksStore, StoreWizardsTowerBooksStore]
|
StoreDieselEbooksStore,
|
||||||
|
StoreEbookscomStore,
|
||||||
|
StoreEPubBuyDEStore,
|
||||||
|
StoreEHarlequinStore,
|
||||||
|
StoreFeedbooksStore,
|
||||||
|
StoreFoylesUKStore,
|
||||||
|
StoreGandalfStore,
|
||||||
|
StoreGoogleBooksStore,
|
||||||
|
StoreGutenbergStore,
|
||||||
|
StoreKoboStore,
|
||||||
|
StoreManyBooksStore,
|
||||||
|
StoreMobileReadStore,
|
||||||
|
StoreNextoStore,
|
||||||
|
StoreOpenLibraryStore,
|
||||||
|
StoreOReillyStore,
|
||||||
|
StorePragmaticBookshelfStore,
|
||||||
|
StoreSmashwordsStore,
|
||||||
|
StoreWaterstonesUKStore,
|
||||||
|
StoreWeightlessBooksStore,
|
||||||
|
StoreWizardsTowerBooksStore
|
||||||
|
]
|
||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
@ -249,6 +249,10 @@ class BarsManager(QObject):
|
|||||||
self.menu_bar = MenuBar(self.location_manager, self.parent())
|
self.menu_bar = MenuBar(self.location_manager, self.parent())
|
||||||
self.parent().setMenuBar(self.menu_bar)
|
self.parent().setMenuBar(self.menu_bar)
|
||||||
|
|
||||||
|
parent.addToolBar(Qt.TopToolBarArea, self.main_bars[0])
|
||||||
|
parent.addToolBar(Qt.BottomToolBarArea, self.main_bars[1])
|
||||||
|
parent.addToolBar(Qt.BottomToolBarArea, self.child_bars[0])
|
||||||
|
|
||||||
self.apply_settings()
|
self.apply_settings()
|
||||||
self.init_bars()
|
self.init_bars()
|
||||||
|
|
||||||
@ -288,12 +292,15 @@ class BarsManager(QObject):
|
|||||||
'''
|
'''
|
||||||
showing_device = self.location_manager.has_device
|
showing_device = self.location_manager.has_device
|
||||||
main_bar = self.main_bars[1 if showing_device else 0]
|
main_bar = self.main_bars[1 if showing_device else 0]
|
||||||
|
hidden_bar = self.main_bars[0 if showing_device else 1]
|
||||||
|
self.parent().addToolBar(Qt.BottomToolBarArea, hidden_bar)
|
||||||
child_bar = self.child_bars[0]
|
child_bar = self.child_bars[0]
|
||||||
for bar in self.bars:
|
for bar in self.bars:
|
||||||
bar.setVisible(False)
|
bar.setVisible(False)
|
||||||
bar.update_lm_actions()
|
bar.update_lm_actions()
|
||||||
if main_bar.added_actions:
|
if main_bar.added_actions:
|
||||||
main_bar.setVisible(True)
|
main_bar.setVisible(True)
|
||||||
|
self.parent().addToolBar(Qt.TopToolBarArea, main_bar)
|
||||||
if child_bar.added_actions:
|
if child_bar.added_actions:
|
||||||
child_bar.setVisible(True)
|
child_bar.setVisible(True)
|
||||||
|
|
||||||
|
@ -259,10 +259,6 @@ class MainWindowMixin(object): # {{{
|
|||||||
self.search_bar = SearchBar(self)
|
self.search_bar = SearchBar(self)
|
||||||
self.bars_manager = BarsManager(self.donate_button,
|
self.bars_manager = BarsManager(self.donate_button,
|
||||||
self.location_manager, self)
|
self.location_manager, self)
|
||||||
for bar in self.bars_manager.main_bars:
|
|
||||||
self.addToolBar(Qt.TopToolBarArea, bar)
|
|
||||||
for bar in self.bars_manager.child_bars:
|
|
||||||
self.addToolBar(Qt.BottomToolBarArea, bar)
|
|
||||||
self.bars_manager.update_bars()
|
self.bars_manager.update_bars()
|
||||||
self.setUnifiedTitleAndToolBarOnMac(True)
|
self.setUnifiedTitleAndToolBarOnMac(True)
|
||||||
|
|
||||||
|
77
src/calibre/gui2/store/gandalf_plugin.py
Normal file
77
src/calibre/gui2/store/gandalf_plugin.py
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from __future__ import (unicode_literals, division, absolute_import, print_function)
|
||||||
|
|
||||||
|
__license__ = 'GPL 3'
|
||||||
|
__copyright__ = '2011, Tomasz Długosz <tomek3d@gmail.com>'
|
||||||
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
|
import re
|
||||||
|
import urllib
|
||||||
|
from contextlib import closing
|
||||||
|
|
||||||
|
from lxml import html
|
||||||
|
|
||||||
|
from PyQt4.Qt import QUrl
|
||||||
|
|
||||||
|
from calibre import browser, url_slash_cleaner
|
||||||
|
from calibre.gui2 import open_url
|
||||||
|
from calibre.gui2.store import StorePlugin
|
||||||
|
from calibre.gui2.store.basic_config import BasicStoreConfig
|
||||||
|
from calibre.gui2.store.search_result import SearchResult
|
||||||
|
from calibre.gui2.store.web_store_dialog import WebStoreDialog
|
||||||
|
|
||||||
|
class GandalfStore(BasicStoreConfig, StorePlugin):
|
||||||
|
|
||||||
|
def open(self, parent=None, detail_item=None, external=False):
|
||||||
|
url = 'http://www.gandalf.com.pl/ebooks/'
|
||||||
|
|
||||||
|
if external or self.config.get('open_external', False):
|
||||||
|
open_url(QUrl(url_slash_cleaner(detail_item if detail_item else url)))
|
||||||
|
else:
|
||||||
|
d = WebStoreDialog(self.gui, url, parent, detail_item)
|
||||||
|
d.setWindowTitle(self.name)
|
||||||
|
d.set_tags(self.config.get('tags', ''))
|
||||||
|
d.exec_()
|
||||||
|
|
||||||
|
def search(self, query, max_results=10, timeout=60):
|
||||||
|
url = 'http://www.gandalf.com.pl/s/'
|
||||||
|
values={
|
||||||
|
'search': query.encode('iso8859_2'),
|
||||||
|
'dzialx':'11'
|
||||||
|
}
|
||||||
|
|
||||||
|
br = browser()
|
||||||
|
|
||||||
|
counter = max_results
|
||||||
|
with closing(br.open(url, data=urllib.urlencode(values), timeout=timeout)) as f:
|
||||||
|
doc = html.fromstring(f.read())
|
||||||
|
for data in doc.xpath('//div[@class="box"]'):
|
||||||
|
if counter <= 0:
|
||||||
|
break
|
||||||
|
|
||||||
|
id = ''.join(data.xpath('.//div[@class="info"]/h3/a/@href'))
|
||||||
|
if not id:
|
||||||
|
continue
|
||||||
|
|
||||||
|
cover_url = ''.join(data.xpath('.//img/@src'))
|
||||||
|
title = ''.join(data.xpath('.//div[@class="info"]/h3/a/@title'))
|
||||||
|
formats = title.split()
|
||||||
|
formats = formats[-1]
|
||||||
|
author = ''.join(data.xpath('.//div[@class="info"]/h4/text() | .//div[@class="info"]/h4/span/text()'))
|
||||||
|
price = ''.join(data.xpath('.//h3[@class="promocja"]/text()'))
|
||||||
|
price = re.sub('PLN', 'zł', price)
|
||||||
|
price = re.sub('\.', ',', price)
|
||||||
|
|
||||||
|
counter -= 1
|
||||||
|
|
||||||
|
s = SearchResult()
|
||||||
|
s.cover_url = cover_url
|
||||||
|
s.title = title.strip()
|
||||||
|
s.author = author.strip()
|
||||||
|
s.price = price
|
||||||
|
s.detail_item = id.strip()
|
||||||
|
s.drm = SearchResult.DRM_UNKNOWN
|
||||||
|
s.formats = formats.upper().strip()
|
||||||
|
|
||||||
|
yield s
|
@ -63,6 +63,7 @@ class NextoStore(BasicStoreConfig, StorePlugin):
|
|||||||
|
|
||||||
cover_url = ''.join(data.xpath('.//img[@class="cover"]/@src'))
|
cover_url = ''.join(data.xpath('.//img[@class="cover"]/@src'))
|
||||||
title = ''.join(data.xpath('.//a[@class="title"]/text()'))
|
title = ''.join(data.xpath('.//a[@class="title"]/text()'))
|
||||||
|
title = re.sub(r' - ebook$', '', title)
|
||||||
formats = ', '.join(data.xpath('.//ul[@class="formats_available"]/li//b/text()'))
|
formats = ', '.join(data.xpath('.//ul[@class="formats_available"]/li//b/text()'))
|
||||||
DrmFree = re.search(r'bez.DRM', formats)
|
DrmFree = re.search(r'bez.DRM', formats)
|
||||||
formats = re.sub(r'\(.+\)', '', formats)
|
formats = re.sub(r'\(.+\)', '', formats)
|
||||||
|
84
src/calibre/gui2/store/pragmatic_bookshelf_plugin.py
Normal file
84
src/calibre/gui2/store/pragmatic_bookshelf_plugin.py
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from __future__ import (unicode_literals, division, absolute_import, print_function)
|
||||||
|
|
||||||
|
__license__ = 'GPL 3'
|
||||||
|
__copyright__ = '2011, John Schember <john@nachtimwald.com>'
|
||||||
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
|
import urllib
|
||||||
|
from contextlib import closing
|
||||||
|
|
||||||
|
from lxml import html
|
||||||
|
|
||||||
|
from PyQt4.Qt import QUrl
|
||||||
|
|
||||||
|
from calibre import browser, url_slash_cleaner
|
||||||
|
from calibre.gui2 import open_url
|
||||||
|
from calibre.gui2.store import StorePlugin
|
||||||
|
from calibre.gui2.store.basic_config import BasicStoreConfig
|
||||||
|
from calibre.gui2.store.search_result import SearchResult
|
||||||
|
from calibre.gui2.store.web_store_dialog import WebStoreDialog
|
||||||
|
|
||||||
|
class PragmaticBookshelfStore(BasicStoreConfig, StorePlugin):
|
||||||
|
|
||||||
|
def open(self, parent=None, detail_item=None, external=False):
|
||||||
|
url = 'http://pragprog.com/'
|
||||||
|
|
||||||
|
if external or self.config.get('open_external', False):
|
||||||
|
open_url(QUrl(url_slash_cleaner(detail_item if detail_item else url)))
|
||||||
|
else:
|
||||||
|
d = WebStoreDialog(self.gui, url, parent, detail_item)
|
||||||
|
d.setWindowTitle(self.name)
|
||||||
|
d.set_tags(self.config.get('tags', ''))
|
||||||
|
d.exec_()
|
||||||
|
|
||||||
|
def search(self, query, max_results=10, timeout=60):
|
||||||
|
'''
|
||||||
|
OPDS based search.
|
||||||
|
|
||||||
|
We really should get the catelog from http://pragprog.com/catalog.opds
|
||||||
|
and look for the application/opensearchdescription+xml entry.
|
||||||
|
Then get the opensearch description to get the search url and
|
||||||
|
format. However, we are going to be lazy and hard code it.
|
||||||
|
'''
|
||||||
|
url = 'http://pragprog.com/catalog/search?q=' + urllib.quote_plus(query)
|
||||||
|
|
||||||
|
br = browser()
|
||||||
|
|
||||||
|
counter = max_results
|
||||||
|
with closing(br.open(url, timeout=timeout)) as f:
|
||||||
|
# Use html instead of etree as html allows us
|
||||||
|
# to ignore the namespace easily.
|
||||||
|
doc = html.fromstring(f.read())
|
||||||
|
for data in doc.xpath('//entry'):
|
||||||
|
if counter <= 0:
|
||||||
|
break
|
||||||
|
|
||||||
|
id = ''.join(data.xpath('.//link[@rel="http://opds-spec.org/acquisition/buy"]/@href'))
|
||||||
|
if not id:
|
||||||
|
continue
|
||||||
|
|
||||||
|
price = ''.join(data.xpath('.//price/@currencycode')).strip()
|
||||||
|
price += ' '
|
||||||
|
price += ''.join(data.xpath('.//price/text()')).strip()
|
||||||
|
if not price.strip():
|
||||||
|
continue
|
||||||
|
|
||||||
|
cover_url = ''.join(data.xpath('.//link[@rel="http://opds-spec.org/cover"]/@href'))
|
||||||
|
|
||||||
|
title = ''.join(data.xpath('.//title/text()'))
|
||||||
|
author = ''.join(data.xpath('.//author//text()'))
|
||||||
|
|
||||||
|
counter -= 1
|
||||||
|
|
||||||
|
s = SearchResult()
|
||||||
|
s.cover_url = cover_url
|
||||||
|
s.title = title.strip()
|
||||||
|
s.author = author.strip()
|
||||||
|
s.price = price.strip()
|
||||||
|
s.detail_item = id.strip()
|
||||||
|
s.drm = SearchResult.DRM_UNLOCKED
|
||||||
|
s.formats = 'EPUB, PDF, MOBI'
|
||||||
|
|
||||||
|
yield s
|
@ -190,7 +190,7 @@ class SearchDialog(QDialog, Ui_Dialog):
|
|||||||
else:
|
else:
|
||||||
self.resize_columns()
|
self.resize_columns()
|
||||||
|
|
||||||
self.open_external.setChecked(self.config.get('open_external', False))
|
self.open_external.setChecked(self.config.get('open_external', True))
|
||||||
|
|
||||||
store_check = self.config.get('store_checked', None)
|
store_check = self.config.get('store_checked', None)
|
||||||
if store_check:
|
if store_check:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user