mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-07-09 03:04:10 -04:00
Merge from trunk
This commit is contained in:
commit
b7eff9ea91
@ -1,4 +1,4 @@
|
|||||||
# vim:fileencoding=UTF-8:ts=2:sw=2:sta:et:sts=2:ai
|
# vim:fileencoding=utf-8:ts=2:sw=2:sta:et:sts=2:ai
|
||||||
# Each release can have new features and bug fixes. Each of which
|
# Each release can have new features and bug fixes. Each of which
|
||||||
# must have a title and can optionally have linked tickets and a description.
|
# must have a title and can optionally have linked tickets and a description.
|
||||||
# In addition they can have a type field which defaults to minor, but should be major
|
# In addition they can have a type field which defaults to minor, but should be major
|
||||||
@ -20,6 +20,66 @@
|
|||||||
# new recipes:
|
# new recipes:
|
||||||
# - title:
|
# - title:
|
||||||
|
|
||||||
|
- version: 0.9.27
|
||||||
|
date: 2013-04-12
|
||||||
|
|
||||||
|
new features:
|
||||||
|
- title: "Metadata download: Add two new sources for covers: Google Image Search and bigbooksearch.com."
|
||||||
|
description: "To enable them go to Preferences->Metadata download and enable the 'Google Image' and 'Big Book Search' sources. Google Images is useful for finding larger covers as well as alternate versions of the cover. Big Book Search searches for alternate covers from amazon.com. It can occasionally find nicer covers than the direct Amazon source. Note that both these sources download multiple covers for a single book. Some of these covers can be wrong (i.e. they may be of a different book or not covers at all, so you should inspect the results and manually pick the best match). When bulk downloading, these sources are only used if the other sources find no covers."
|
||||||
|
type: major
|
||||||
|
|
||||||
|
- title: "Content server: Allow specifying a restriction to use for the server when embedding it as a WSGI app."
|
||||||
|
tickets: [1167951]
|
||||||
|
|
||||||
|
- title: "Get Books: Add a plugin for the Koobe Polish book store"
|
||||||
|
|
||||||
|
- title: "calibredb add_format: Add an option to not replace existing formats. Also pep8 compliance."
|
||||||
|
|
||||||
|
- title: "Allow restoring of the ORIGINAL_XXX format by right-clicking it in the book details panel"
|
||||||
|
|
||||||
|
bug fixes:
|
||||||
|
- title: "AZW3 Input: Do not fail to identify JPEG images with 8BIM headers created with Adobe Photoshop."
|
||||||
|
tickets: [1167985]
|
||||||
|
|
||||||
|
- title: "Amazon metadata download: Ignore Spanish edition entries when searching for a book on amazon.com"
|
||||||
|
|
||||||
|
- title: "TXT Input: When converting a txt file with a Byte Order Mark, remove the Byte Order Mark before further processing as it can cause the first line of the text to be mis-interpreted."
|
||||||
|
|
||||||
|
- title: "Get Books: Fix searching for current book/title/author by right clicking the get books icon"
|
||||||
|
|
||||||
|
- title: "Get Books: Update nexto, gutenberg, and virtualo store plugins for website changes"
|
||||||
|
|
||||||
|
- title: "Amazon metadata download: When downloading from amazon.co.jp handle the 'Black curtain redirect' for adult titles."
|
||||||
|
tickets: [1165628]
|
||||||
|
|
||||||
|
- title: "When extracting zip files do not allow maliciously created zip files to overwrite other files on the system"
|
||||||
|
|
||||||
|
- title: "RTF Input: Handle RTF files with invalid border style specifications"
|
||||||
|
tickets: [1021270]
|
||||||
|
|
||||||
|
improved recipes:
|
||||||
|
- The Escapist
|
||||||
|
- San Francisco Chronicle
|
||||||
|
- The Onion
|
||||||
|
- Fronda
|
||||||
|
- Tom's Hardware
|
||||||
|
- New Yorker
|
||||||
|
- Financial Times UK
|
||||||
|
- Business Week Magazine
|
||||||
|
- Victoria Times
|
||||||
|
- tvxs
|
||||||
|
- The Independent
|
||||||
|
|
||||||
|
new recipes:
|
||||||
|
- title: Economia
|
||||||
|
author: Manish Bhattarai
|
||||||
|
|
||||||
|
- title: Universe Today
|
||||||
|
author: seird
|
||||||
|
|
||||||
|
- title: The Galaxy's Edge
|
||||||
|
author: Krittika Goyal
|
||||||
|
|
||||||
- version: 0.9.26
|
- version: 0.9.26
|
||||||
date: 2013-04-05
|
date: 2013-04-05
|
||||||
|
|
||||||
|
@ -436,8 +436,8 @@ generate a Table of Contents in the converted ebook, based on the actual content
|
|||||||
|
|
||||||
.. note:: Using these options can be a little challenging to get exactly right.
|
.. note:: Using these options can be a little challenging to get exactly right.
|
||||||
If you prefer creating/editing the Table of Contents by hand, convert to
|
If you prefer creating/editing the Table of Contents by hand, convert to
|
||||||
the EPUB or AZW3 formats and select the checkbox at the bottom of the
|
the EPUB or AZW3 formats and select the checkbox at the bottom of the Table
|
||||||
screen that says
|
of Contents section of the conversion dialog that says
|
||||||
:guilabel:`Manually fine-tune the Table of Contents after conversion`.
|
:guilabel:`Manually fine-tune the Table of Contents after conversion`.
|
||||||
This will launch the ToC Editor tool after the conversion. It allows you to
|
This will launch the ToC Editor tool after the conversion. It allows you to
|
||||||
create entries in the Table of Contents by simply clicking the place in the
|
create entries in the Table of Contents by simply clicking the place in the
|
||||||
|
@ -802,6 +802,12 @@ Downloading from the Internet can sometimes result in a corrupted download. If t
|
|||||||
* Try temporarily disabling your antivirus program (Microsoft Security Essentials, or Kaspersky or Norton or McAfee or whatever). This is most likely the culprit if the upgrade process is hanging in the middle.
|
* Try temporarily disabling your antivirus program (Microsoft Security Essentials, or Kaspersky or Norton or McAfee or whatever). This is most likely the culprit if the upgrade process is hanging in the middle.
|
||||||
* Try rebooting your computer and running a registry cleaner like `Wise registry cleaner <http://www.wisecleaner.com>`_.
|
* Try rebooting your computer and running a registry cleaner like `Wise registry cleaner <http://www.wisecleaner.com>`_.
|
||||||
* Try downloading the installer with an alternate browser. For example if you are using Internet Explorer, try using Firefox or Chrome instead.
|
* Try downloading the installer with an alternate browser. For example if you are using Internet Explorer, try using Firefox or Chrome instead.
|
||||||
|
* If you get an error about a missing DLL on windows, then most likely, the
|
||||||
|
permissions on your temporary folder are incorrect. Go to the folder
|
||||||
|
:file:`C:\\Users\\USERNAME\\AppData\\Local` in Windows explorer and then
|
||||||
|
right click on the :file:`Temp` folder and select :guilabel:`Properties` and go to
|
||||||
|
the :guilabel:`Security` tab. Make sure that your user account has full control
|
||||||
|
for this folder.
|
||||||
|
|
||||||
If you still cannot get the installer to work and you are on windows, you can use the `calibre portable install <http://calibre-ebook.com/download_portable>`_, which does not need an installer (it is just a zip file).
|
If you still cannot get the installer to work and you are on windows, you can use the `calibre portable install <http://calibre-ebook.com/download_portable>`_, which does not need an installer (it is just a zip file).
|
||||||
|
|
||||||
|
@ -367,6 +367,8 @@ For example::
|
|||||||
|
|
||||||
date:>10daysago
|
date:>10daysago
|
||||||
date:<=45daysago
|
date:<=45daysago
|
||||||
|
|
||||||
|
To avoid potential problems with translated strings when using a non-English version of calibre, the strings ``_today``, ``_yesterday``, ``_thismonth``, and ``_daysago`` are always available. They are not translated.
|
||||||
|
|
||||||
You can search for books that have a format of a certain size like this::
|
You can search for books that have a format of a certain size like this::
|
||||||
|
|
||||||
|
@ -91,7 +91,11 @@ First, we have to create a WSGI *adapter* for the calibre content server. Here i
|
|||||||
# Path to the calibre library to be served
|
# Path to the calibre library to be served
|
||||||
# The server process must have write permission for all files/dirs
|
# The server process must have write permission for all files/dirs
|
||||||
# in this directory or BAD things will happen
|
# in this directory or BAD things will happen
|
||||||
path_to_library='/home/kovid/documents/demo library'
|
path_to_library='/home/kovid/documents/demo library',
|
||||||
|
|
||||||
|
# The virtual library (restriction) to be used when serving this
|
||||||
|
# library.
|
||||||
|
virtual_library=None
|
||||||
)
|
)
|
||||||
|
|
||||||
del create_wsgi_app
|
del create_wsgi_app
|
||||||
|
@ -11,22 +11,22 @@ from calibre.web.feeds.news import BasicNewsRecipe
|
|||||||
class EcoGeek(BasicNewsRecipe):
|
class EcoGeek(BasicNewsRecipe):
|
||||||
title = 'EcoGeek'
|
title = 'EcoGeek'
|
||||||
__author__ = 'Darko Miletic'
|
__author__ = 'Darko Miletic'
|
||||||
description = 'EcoGeek - Technology for the Environment Blog Feed'
|
description = 'EcoGeek - Technology for the Environment Blog Feed'
|
||||||
publisher = 'EcoGeek'
|
publisher = 'EcoGeek'
|
||||||
language = 'en'
|
language = 'en'
|
||||||
|
|
||||||
category = 'news, ecology, blog'
|
category = 'news, ecology, blog'
|
||||||
oldest_article = 7
|
oldest_article = 30
|
||||||
max_articles_per_feed = 100
|
max_articles_per_feed = 100
|
||||||
no_stylesheets = True
|
no_stylesheets = True
|
||||||
use_embedded_content = True
|
use_embedded_content = True
|
||||||
|
|
||||||
html2lrf_options = [
|
html2lrf_options = [
|
||||||
'--comment', description
|
'--comment', description
|
||||||
, '--category', category
|
, '--category', category
|
||||||
, '--publisher', publisher
|
, '--publisher', publisher
|
||||||
]
|
]
|
||||||
|
|
||||||
html2epub_options = 'publisher="' + publisher + '"\ncomments="' + description + '"\ntags="' + category + '"'
|
|
||||||
|
|
||||||
|
html2epub_options = 'publisher="' + publisher + '"\ncomments="' + description + '"\ntags="' + category + '"'
|
||||||
feeds = [(u'Posts', u'http://feeds2.feedburner.com/EcoGeek')]
|
feeds = [(u'Posts', u'http://feeds2.feedburner.com/EcoGeek')]
|
||||||
|
|
||||||
|
@ -1,90 +0,0 @@
|
|||||||
import re
|
|
||||||
from calibre.web.feeds.news import BasicNewsRecipe
|
|
||||||
|
|
||||||
class GiveMeSomethingToRead(BasicNewsRecipe):
|
|
||||||
title = u'Give Me Something To Read'
|
|
||||||
description = 'Curation / aggregation of articles on diverse topics'
|
|
||||||
language = 'en'
|
|
||||||
__author__ = 'barty on mobileread.com forum'
|
|
||||||
max_articles_per_feed = 100
|
|
||||||
no_stylesheets = False
|
|
||||||
timefmt = ' [%a, %d %b, %Y]'
|
|
||||||
oldest_article = 365
|
|
||||||
auto_cleanup = True
|
|
||||||
INDEX = 'http://givemesomethingtoread.com'
|
|
||||||
CATEGORIES = [
|
|
||||||
# comment out categories you don't want
|
|
||||||
# (user friendly name, system name, max number of articles to load)
|
|
||||||
('The Arts','arts',25),
|
|
||||||
('Science','science',30),
|
|
||||||
('Technology','technology',30),
|
|
||||||
('Politics','politics',20),
|
|
||||||
('Media','media',30),
|
|
||||||
('Crime','crime',15),
|
|
||||||
('Other articles','',10)
|
|
||||||
]
|
|
||||||
|
|
||||||
def parse_index(self):
|
|
||||||
self.cover_url = 'http://thegretchenshow.files.wordpress.com/2009/12/well-read-cat-small.jpg'
|
|
||||||
feeds = []
|
|
||||||
seen_urls = set([])
|
|
||||||
regex = re.compile( r'http://(www\.)?([^/:]+)', re.I)
|
|
||||||
|
|
||||||
for category in self.CATEGORIES:
|
|
||||||
|
|
||||||
(cat_name, tag, max_articles) = category
|
|
||||||
|
|
||||||
tagurl = '' if tag=='' else '/tagged/'+tag
|
|
||||||
self.log('Reading category:', cat_name)
|
|
||||||
|
|
||||||
articles = []
|
|
||||||
pageno = 1
|
|
||||||
|
|
||||||
while len(articles) < max_articles and pageno < 100:
|
|
||||||
|
|
||||||
page = "%s%s/page/%d" % (self.INDEX, tagurl, pageno) if pageno > 1 else self.INDEX + tagurl
|
|
||||||
pageno += 1
|
|
||||||
|
|
||||||
self.log('\tReading page:', page)
|
|
||||||
try:
|
|
||||||
soup = self.index_to_soup(page)
|
|
||||||
except:
|
|
||||||
break
|
|
||||||
|
|
||||||
headers = soup.findAll('h2')
|
|
||||||
if len(headers) == .0:
|
|
||||||
break
|
|
||||||
|
|
||||||
for header in headers:
|
|
||||||
atag = header.find('a')
|
|
||||||
url = atag['href']
|
|
||||||
# skip promotionals and duplicate
|
|
||||||
if url.startswith('http://givemesomethingtoread') or url.startswith('/') or url in seen_urls:
|
|
||||||
continue
|
|
||||||
seen_urls.add(url)
|
|
||||||
title = self.tag_to_string(header)
|
|
||||||
self.log('\tFound article:', title)
|
|
||||||
#self.log('\t', url)
|
|
||||||
desc = header.parent.find('blockquote')
|
|
||||||
desc = self.tag_to_string(desc) if desc else ''
|
|
||||||
m = regex.match( url)
|
|
||||||
if m:
|
|
||||||
desc = "[%s] %s" % (m.group(2), desc)
|
|
||||||
#self.log('\t', desc)
|
|
||||||
date = ''
|
|
||||||
p = header.parent.previousSibling
|
|
||||||
# navigate up to find h3, which contains the date
|
|
||||||
while p:
|
|
||||||
if hasattr(p,'name') and p.name == 'h3':
|
|
||||||
date = self.tag_to_string(p)
|
|
||||||
break
|
|
||||||
p = p.previousSibling
|
|
||||||
articles.append({'title':title,'url':url,'description':desc,'date':date})
|
|
||||||
if len(articles) >= max_articles:
|
|
||||||
break
|
|
||||||
|
|
||||||
if articles:
|
|
||||||
feeds.append((cat_name, articles))
|
|
||||||
|
|
||||||
return feeds
|
|
||||||
|
|
@ -1,448 +1,229 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
|
|
||||||
__license__ = 'GPL v3'
|
|
||||||
__copyright__ = 'Copyright 2010 Starson17'
|
|
||||||
'''
|
|
||||||
www.gocomics.com
|
|
||||||
'''
|
|
||||||
from calibre.web.feeds.news import BasicNewsRecipe
|
from calibre.web.feeds.news import BasicNewsRecipe
|
||||||
import mechanize, re
|
|
||||||
|
|
||||||
class GoComics(BasicNewsRecipe):
|
|
||||||
title = 'GoComics'
|
class Comics(BasicNewsRecipe):
|
||||||
|
title = 'Comics.com'
|
||||||
__author__ = 'Starson17'
|
__author__ = 'Starson17'
|
||||||
__version__ = '1.06'
|
description = 'Comics from comics.com. You should customize this recipe to fetch only the comics you are interested in'
|
||||||
__date__ = '07 June 2011'
|
|
||||||
description = u'200+ Comics - Customize for more days/comics: Defaults to 7 days, 25 comics - 20 general, 5 editorial.'
|
|
||||||
category = 'news, comics'
|
|
||||||
language = 'en'
|
language = 'en'
|
||||||
use_embedded_content= False
|
use_embedded_content= False
|
||||||
no_stylesheets = True
|
no_stylesheets = True
|
||||||
|
oldest_article = 24
|
||||||
remove_javascript = True
|
remove_javascript = True
|
||||||
cover_url = 'http://paulbuckley14059.files.wordpress.com/2008/06/calvin-and-hobbes.jpg'
|
cover_url = 'http://www.bsb.lib.tx.us/images/comics.com.gif'
|
||||||
remove_attributes = ['style']
|
recursions = 0
|
||||||
|
max_articles_per_feed = 10
|
||||||
####### USER PREFERENCES - COMICS, IMAGE SIZE AND NUMBER OF COMICS TO RETRIEVE ########
|
|
||||||
# num_comics_to_get - I've tried up to 99 on Calvin&Hobbes
|
|
||||||
num_comics_to_get = 7
|
num_comics_to_get = 7
|
||||||
# comic_size 300 is small, 600 is medium, 900 is large, 1500 is extra-large
|
simultaneous_downloads = 1
|
||||||
comic_size = 900
|
# delay = 3
|
||||||
# CHOOSE COMIC STRIPS BELOW - REMOVE COMMENT '# ' FROM IN FRONT OF DESIRED STRIPS
|
|
||||||
# Please do not overload their servers by selecting all comics and 1000 strips from each!
|
|
||||||
|
|
||||||
conversion_options = {'linearize_tables' : True
|
keep_only_tags = [dict(name='h1'),
|
||||||
, 'comment' : description
|
dict(name='p', attrs={'class':'feature_item'})
|
||||||
, 'tags' : category
|
|
||||||
, 'language' : language
|
|
||||||
}
|
|
||||||
|
|
||||||
keep_only_tags = [dict(name='div', attrs={'class':['feature','banner']}),
|
|
||||||
]
|
]
|
||||||
|
|
||||||
remove_tags = [dict(name='a', attrs={'class':['beginning','prev','cal','next','newest']}),
|
|
||||||
dict(name='div', attrs={'class':['tag-wrapper']}),
|
|
||||||
dict(name='a', attrs={'href':re.compile(r'.*mutable_[0-9]+', re.IGNORECASE)}),
|
|
||||||
dict(name='img', attrs={'src':re.compile(r'.*mutable_[0-9]+', re.IGNORECASE)}),
|
|
||||||
dict(name='ul', attrs={'class':['share-nav','feature-nav']}),
|
|
||||||
]
|
|
||||||
|
|
||||||
def get_browser(self):
|
|
||||||
br = BasicNewsRecipe.get_browser(self)
|
|
||||||
cookies = mechanize.CookieJar()
|
|
||||||
br = mechanize.build_opener(mechanize.HTTPCookieProcessor(cookies))
|
|
||||||
br.addheaders = [('Referer','http://www.gocomics.com/')]
|
|
||||||
return br
|
|
||||||
|
|
||||||
def parse_index(self):
|
def parse_index(self):
|
||||||
feeds = []
|
feeds = []
|
||||||
for title, url in [
|
for title, url in [
|
||||||
(u"2 Cows and a Chicken", u"http://www.gocomics.com/2cowsandachicken"),
|
("9 Chickweed Lane", "http://gocomics.com/9_chickweed_lane"),
|
||||||
#(u"9 Chickweed Lane", u"http://www.gocomics.com/9chickweedlane"),
|
("Agnes", "http://gocomics.com/agnes"),
|
||||||
(u"9 to 5", u"http://www.gocomics.com/9to5"),
|
("Alley Oop", "http://gocomics.com/alley_oop"),
|
||||||
#(u"Adam At Home", u"http://www.gocomics.com/adamathome"),
|
("Andy Capp", "http://gocomics.com/andy_capp"),
|
||||||
(u"Agnes", u"http://www.gocomics.com/agnes"),
|
("Arlo & Janis", "http://gocomics.com/arlo&janis"),
|
||||||
#(u"Alley Oop", u"http://www.gocomics.com/alleyoop"),
|
("B.C.", "http://gocomics.com/bc"),
|
||||||
#(u"Andy Capp", u"http://www.gocomics.com/andycapp"),
|
("Ballard Street", "http://gocomics.com/ballard_street"),
|
||||||
#(u"Animal Crackers", u"http://www.gocomics.com/animalcrackers"),
|
# ("Ben", "http://comics.com/ben"),
|
||||||
#(u"Annie", u"http://www.gocomics.com/annie"),
|
# ("Betty", "http://comics.com/betty"),
|
||||||
#(u"Arlo & Janis", u"http://www.gocomics.com/arloandjanis"),
|
# ("Big Nate", "http://comics.com/big_nate"),
|
||||||
#(u"Ask Shagg", u"http://www.gocomics.com/askshagg"),
|
# ("Brevity", "http://comics.com/brevity"),
|
||||||
(u"B.C.", u"http://www.gocomics.com/bc"),
|
# ("Candorville", "http://comics.com/candorville"),
|
||||||
#(u"Back in the Day", u"http://www.gocomics.com/backintheday"),
|
# ("Cheap Thrills", "http://comics.com/cheap_thrills"),
|
||||||
#(u"Bad Reporter", u"http://www.gocomics.com/badreporter"),
|
# ("Committed", "http://comics.com/committed"),
|
||||||
#(u"Baldo", u"http://www.gocomics.com/baldo"),
|
# ("Cow & Boy", "http://comics.com/cow&boy"),
|
||||||
#(u"Ballard Street", u"http://www.gocomics.com/ballardstreet"),
|
# ("Daddy's Home", "http://comics.com/daddys_home"),
|
||||||
#(u"Barkeater Lake", u"http://www.gocomics.com/barkeaterlake"),
|
# ("Dog eat Doug", "http://comics.com/dog_eat_doug"),
|
||||||
#(u"Basic Instructions", u"http://www.gocomics.com/basicinstructions"),
|
# ("Drabble", "http://comics.com/drabble"),
|
||||||
#(u"Ben", u"http://www.gocomics.com/ben"),
|
# ("F Minus", "http://comics.com/f_minus"),
|
||||||
#(u"Betty", u"http://www.gocomics.com/betty"),
|
# ("Family Tree", "http://comics.com/family_tree"),
|
||||||
#(u"Bewley", u"http://www.gocomics.com/bewley"),
|
# ("Farcus", "http://comics.com/farcus"),
|
||||||
#(u"Big Nate", u"http://www.gocomics.com/bignate"),
|
# ("Fat Cats Classics", "http://comics.com/fat_cats_classics"),
|
||||||
#(u"Big Top", u"http://www.gocomics.com/bigtop"),
|
# ("Ferd'nand", "http://comics.com/ferdnand"),
|
||||||
#(u"Biographic", u"http://www.gocomics.com/biographic"),
|
# ("Flight Deck", "http://comics.com/flight_deck"),
|
||||||
#(u"Birdbrains", u"http://www.gocomics.com/birdbrains"),
|
# ("Flo & Friends", "http://comics.com/flo&friends"),
|
||||||
#(u"Bleeker: The Rechargeable Dog", u"http://www.gocomics.com/bleeker"),
|
# ("Fort Knox", "http://comics.com/fort_knox"),
|
||||||
#(u"Bliss", u"http://www.gocomics.com/bliss"),
|
# ("Frank & Ernest", "http://comics.com/frank&ernest"),
|
||||||
(u"Bloom County", u"http://www.gocomics.com/bloomcounty"),
|
# ("Frazz", "http://comics.com/frazz"),
|
||||||
#(u"Bo Nanas", u"http://www.gocomics.com/bonanas"),
|
# ("Free Range", "http://comics.com/free_range"),
|
||||||
#(u"Bob the Squirrel", u"http://www.gocomics.com/bobthesquirrel"),
|
# ("Geech Classics", "http://comics.com/geech_classics"),
|
||||||
#(u"Boomerangs", u"http://www.gocomics.com/boomerangs"),
|
# ("Get Fuzzy", "http://comics.com/get_fuzzy"),
|
||||||
#(u"Bottomliners", u"http://www.gocomics.com/bottomliners"),
|
# ("Girls & Sports", "http://comics.com/girls&sports"),
|
||||||
#(u"Bound and Gagged", u"http://www.gocomics.com/boundandgagged"),
|
# ("Graffiti", "http://comics.com/graffiti"),
|
||||||
#(u"Brainwaves", u"http://www.gocomics.com/brainwaves"),
|
# ("Grand Avenue", "http://comics.com/grand_avenue"),
|
||||||
#(u"Brenda Starr", u"http://www.gocomics.com/brendastarr"),
|
# ("Heathcliff", "http://comics.com/heathcliff"),
|
||||||
#(u"Brevity", u"http://www.gocomics.com/brevity"),
|
# "Heathcliff, a street-smart and mischievous cat with many adventures."
|
||||||
#(u"Brewster Rockit", u"http://www.gocomics.com/brewsterrockit"),
|
# ("Herb and Jamaal", "http://comics.com/herb_and_jamaal"),
|
||||||
#(u"Broom Hilda", u"http://www.gocomics.com/broomhilda"),
|
# ("Herman", "http://comics.com/herman"),
|
||||||
(u"Calvin and Hobbes", u"http://www.gocomics.com/calvinandhobbes"),
|
# ("Home and Away", "http://comics.com/home_and_away"),
|
||||||
#(u"Candorville", u"http://www.gocomics.com/candorville"),
|
# ("It's All About You", "http://comics.com/its_all_about_you"),
|
||||||
#(u"Cathy", u"http://www.gocomics.com/cathy"),
|
# ("Jane's World", "http://comics.com/janes_world"),
|
||||||
#(u"C'est la Vie", u"http://www.gocomics.com/cestlavie"),
|
# ("Jump Start", "http://comics.com/jump_start"),
|
||||||
#(u"Cheap Thrills", u"http://www.gocomics.com/cheapthrills"),
|
# ("Kit 'N' Carlyle", "http://comics.com/kit_n_carlyle"),
|
||||||
#(u"Chuckle Bros", u"http://www.gocomics.com/chucklebros"),
|
# ("Li'l Abner Classics", "http://comics.com/lil_abner_classics"),
|
||||||
#(u"Citizen Dog", u"http://www.gocomics.com/citizendog"),
|
# ("Liberty Meadows", "http://comics.com/liberty_meadows"),
|
||||||
#(u"Cleats", u"http://www.gocomics.com/cleats"),
|
# ("Little Dog Lost", "http://comics.com/little_dog_lost"),
|
||||||
#(u"Close to Home", u"http://www.gocomics.com/closetohome"),
|
# ("Lola", "http://comics.com/lola"),
|
||||||
#(u"Committed", u"http://www.gocomics.com/committed"),
|
# ("Luann", "http://comics.com/luann"),
|
||||||
#(u"Compu-toon", u"http://www.gocomics.com/compu-toon"),
|
# ("Marmaduke", "http://comics.com/marmaduke"),
|
||||||
#(u"Cornered", u"http://www.gocomics.com/cornered"),
|
# ("Meg! Classics", "http://comics.com/meg_classics"),
|
||||||
#(u"Cow & Boy", u"http://www.gocomics.com/cow&boy"),
|
# ("Minimum Security", "http://comics.com/minimum_security"),
|
||||||
#(u"Cul de Sac", u"http://www.gocomics.com/culdesac"),
|
# ("Moderately Confused", "http://comics.com/moderately_confused"),
|
||||||
#(u"Daddy's Home", u"http://www.gocomics.com/daddyshome"),
|
# ("Momma", "http://comics.com/momma"),
|
||||||
#(u"Deep Cover", u"http://www.gocomics.com/deepcover"),
|
# ("Monty", "http://comics.com/monty"),
|
||||||
#(u"Dick Tracy", u"http://www.gocomics.com/dicktracy"),
|
# ("Motley Classics", "http://comics.com/motley_classics"),
|
||||||
(u"Dog Eat Doug", u"http://www.gocomics.com/dogeatdoug"),
|
# ("Nancy", "http://comics.com/nancy"),
|
||||||
#(u"Domestic Abuse", u"http://www.gocomics.com/domesticabuse"),
|
# ("Natural Selection", "http://comics.com/natural_selection"),
|
||||||
(u"Doodles", u"http://www.gocomics.com/doodles"),
|
# ("Nest Heads", "http://comics.com/nest_heads"),
|
||||||
(u"Doonesbury", u"http://www.gocomics.com/doonesbury"),
|
# ("Off The Mark", "http://comics.com/off_the_mark"),
|
||||||
#(u"Drabble", u"http://www.gocomics.com/drabble"),
|
# ("On a Claire Day", "http://comics.com/on_a_claire_day"),
|
||||||
#(u"Eek!", u"http://www.gocomics.com/eek"),
|
# ("One Big Happy Classics", "http://comics.com/one_big_happy_classics"),
|
||||||
#(u"F Minus", u"http://www.gocomics.com/fminus"),
|
# ("Over the Hedge", "http://comics.com/over_the_hedge"),
|
||||||
#(u"Family Tree", u"http://www.gocomics.com/familytree"),
|
# ("PC and Pixel", "http://comics.com/pc_and_pixel"),
|
||||||
#(u"Farcus", u"http://www.gocomics.com/farcus"),
|
# ("Peanuts", "http://comics.com/peanuts"),
|
||||||
(u"Fat Cats Classics", u"http://www.gocomics.com/fatcatsclassics"),
|
# ("Pearls Before Swine", "http://comics.com/pearls_before_swine"),
|
||||||
#(u"Ferd'nand", u"http://www.gocomics.com/ferdnand"),
|
# ("Pickles", "http://comics.com/pickles"),
|
||||||
#(u"Flight Deck", u"http://www.gocomics.com/flightdeck"),
|
# ("Prickly City", "http://comics.com/prickly_city"),
|
||||||
(u"Flo and Friends", u"http://www.gocomics.com/floandfriends"),
|
# ("Raising Duncan Classics", "http://comics.com/raising_duncan_classics"),
|
||||||
#(u"For Better or For Worse", u"http://www.gocomics.com/forbetterorforworse"),
|
# ("Reality Check", "http://comics.com/reality_check"),
|
||||||
#(u"For Heaven's Sake", u"http://www.gocomics.com/forheavenssake"),
|
# ("Red & Rover", "http://comics.com/red&rover"),
|
||||||
#(u"Fort Knox", u"http://www.gocomics.com/fortknox"),
|
# ("Rip Haywire", "http://comics.com/rip_haywire"),
|
||||||
#(u"FoxTrot Classics", u"http://www.gocomics.com/foxtrotclassics"),
|
# ("Ripley's Believe It or Not!", "http://comics.com/ripleys_believe_it_or_not"),
|
||||||
(u"FoxTrot", u"http://www.gocomics.com/foxtrot"),
|
# ("Rose Is Rose", "http://comics.com/rose_is_rose"),
|
||||||
#(u"Frank & Ernest", u"http://www.gocomics.com/frankandernest"),
|
# ("Rubes", "http://comics.com/rubes"),
|
||||||
#(u"Frazz", u"http://www.gocomics.com/frazz"),
|
# ("Rudy Park", "http://comics.com/rudy_park"),
|
||||||
#(u"Fred Basset", u"http://www.gocomics.com/fredbasset"),
|
# ("Scary Gary", "http://comics.com/scary_gary"),
|
||||||
#(u"Free Range", u"http://www.gocomics.com/freerange"),
|
# ("Shirley and Son Classics", "http://comics.com/shirley_and_son_classics"),
|
||||||
#(u"Frog Applause", u"http://www.gocomics.com/frogapplause"),
|
# ("Soup To Nutz", "http://comics.com/soup_to_nutz"),
|
||||||
#(u"Garfield Minus Garfield", u"http://www.gocomics.com/garfieldminusgarfield"),
|
# ("Speed Bump", "http://comics.com/speed_bump"),
|
||||||
(u"Garfield", u"http://www.gocomics.com/garfield"),
|
# ("Spot The Frog", "http://comics.com/spot_the_frog"),
|
||||||
#(u"Gasoline Alley", u"http://www.gocomics.com/gasolinealley"),
|
# ("State of the Union", "http://comics.com/state_of_the_union"),
|
||||||
#(u"Geech Classics", u"http://www.gocomics.com/geechclassics"),
|
# ("Strange Brew", "http://comics.com/strange_brew"),
|
||||||
#(u"Get Fuzzy", u"http://www.gocomics.com/getfuzzy"),
|
# ("Tarzan Classics", "http://comics.com/tarzan_classics"),
|
||||||
#(u"Gil Thorp", u"http://www.gocomics.com/gilthorp"),
|
# ("That's Life", "http://comics.com/thats_life"),
|
||||||
#(u"Ginger Meggs", u"http://www.gocomics.com/gingermeggs"),
|
# ("The Barn", "http://comics.com/the_barn"),
|
||||||
#(u"Girls & Sports", u"http://www.gocomics.com/girlsandsports"),
|
# ("The Born Loser", "http://comics.com/the_born_loser"),
|
||||||
#(u"Graffiti", u"http://www.gocomics.com/graffiti"),
|
# ("The Buckets", "http://comics.com/the_buckets"),
|
||||||
#(u"Grand Avenue", u"http://www.gocomics.com/grandavenue"),
|
# ("The Dinette Set", "http://comics.com/the_dinette_set"),
|
||||||
#(u"Haiku Ewe", u"http://www.gocomics.com/haikuewe"),
|
# ("The Grizzwells", "http://comics.com/the_grizzwells"),
|
||||||
#(u"Heart of the City", u"http://www.gocomics.com/heartofthecity"),
|
# ("The Humble Stumble", "http://comics.com/the_humble_stumble"),
|
||||||
(u"Heathcliff", u"http://www.gocomics.com/heathcliff"),
|
# ("The Knight Life", "http://comics.com/the_knight_life"),
|
||||||
#(u"Herb and Jamaal", u"http://www.gocomics.com/herbandjamaal"),
|
# ("The Meaning of Lila", "http://comics.com/the_meaning_of_lila"),
|
||||||
#(u"Herman", u"http://www.gocomics.com/herman"),
|
# ("The Other Coast", "http://comics.com/the_other_coast"),
|
||||||
#(u"Home and Away", u"http://www.gocomics.com/homeandaway"),
|
# ("The Sunshine Club", "http://comics.com/the_sunshine_club"),
|
||||||
#(u"Housebroken", u"http://www.gocomics.com/housebroken"),
|
# ("Unstrange Phenomena", "http://comics.com/unstrange_phenomena"),
|
||||||
#(u"Hubert and Abby", u"http://www.gocomics.com/hubertandabby"),
|
# ("Watch Your Head", "http://comics.com/watch_your_head"),
|
||||||
#(u"Imagine This", u"http://www.gocomics.com/imaginethis"),
|
# ("Wizard of Id", "http://comics.com/wizard_of_id"),
|
||||||
#(u"In the Bleachers", u"http://www.gocomics.com/inthebleachers"),
|
# ("Working Daze", "http://comics.com/working_daze"),
|
||||||
#(u"In the Sticks", u"http://www.gocomics.com/inthesticks"),
|
# ("Working It Out", "http://comics.com/working_it_out"),
|
||||||
#(u"Ink Pen", u"http://www.gocomics.com/inkpen"),
|
# ("Zack Hill", "http://comics.com/zack_hill"),
|
||||||
#(u"It's All About You", u"http://www.gocomics.com/itsallaboutyou"),
|
# ("(Th)ink", "http://comics.com/think"),
|
||||||
#(u"Jane's World", u"http://www.gocomics.com/janesworld"),
|
# "Tackling the political and social issues impacting communities of color."
|
||||||
#(u"Joe Vanilla", u"http://www.gocomics.com/joevanilla"),
|
# ("Adam Zyglis", "http://comics.com/adam_zyglis"),
|
||||||
#(u"Jump Start", u"http://www.gocomics.com/jumpstart"),
|
# "Known for his excellent caricatures, as well as independent and incisive imagery. "
|
||||||
#(u"Kit 'N' Carlyle", u"http://www.gocomics.com/kitandcarlyle"),
|
# ("Andy Singer", "http://comics.com/andy_singer"),
|
||||||
#(u"La Cucaracha", u"http://www.gocomics.com/lacucaracha"),
|
# ("Bill Day", "http://comics.com/bill_day"),
|
||||||
#(u"Last Kiss", u"http://www.gocomics.com/lastkiss"),
|
# "Powerful images on sensitive issues."
|
||||||
#(u"Legend of Bill", u"http://www.gocomics.com/legendofbill"),
|
# ("Bill Schorr", "http://comics.com/bill_schorr"),
|
||||||
#(u"Liberty Meadows", u"http://www.gocomics.com/libertymeadows"),
|
# ("Bob Englehart", "http://comics.com/bob_englehart"),
|
||||||
#(u"Li'l Abner Classics", u"http://www.gocomics.com/lilabnerclassics"),
|
# ("Brian Fairrington", "http://comics.com/brian_fairrington"),
|
||||||
#(u"Lio", u"http://www.gocomics.com/lio"),
|
# ("Bruce Beattie", "http://comics.com/bruce_beattie"),
|
||||||
#(u"Little Dog Lost", u"http://www.gocomics.com/littledoglost"),
|
# ("Cam Cardow", "http://comics.com/cam_cardow"),
|
||||||
#(u"Little Otto", u"http://www.gocomics.com/littleotto"),
|
# ("Chip Bok", "http://comics.com/chip_bok"),
|
||||||
#(u"Lola", u"http://www.gocomics.com/lola"),
|
# ("Chris Britt", "http://comics.com/chris_britt"),
|
||||||
#(u"Loose Parts", u"http://www.gocomics.com/looseparts"),
|
# ("Chuck Asay", "http://comics.com/chuck_asay"),
|
||||||
#(u"Love Is...", u"http://www.gocomics.com/loveis"),
|
# ("Clay Bennett", "http://comics.com/clay_bennett"),
|
||||||
#(u"Luann", u"http://www.gocomics.com/luann"),
|
# ("Daryl Cagle", "http://comics.com/daryl_cagle"),
|
||||||
#(u"Maintaining", u"http://www.gocomics.com/maintaining"),
|
# ("David Fitzsimmons", "http://comics.com/david_fitzsimmons"),
|
||||||
(u"Marmaduke", u"http://www.gocomics.com/marmaduke"),
|
# "David Fitzsimmons is a new editorial cartoons on comics.com. He is also a staff writer and editorial cartoonist for the Arizona Daily Star. "
|
||||||
#(u"Meg! Classics", u"http://www.gocomics.com/megclassics"),
|
# ("Drew Litton", "http://comics.com/drew_litton"),
|
||||||
#(u"Middle-Aged White Guy", u"http://www.gocomics.com/middleagedwhiteguy"),
|
# "Drew Litton is an artist who is probably best known for his sports cartoons. He received the National Cartoonist Society Sports Cartoon Award for 1993. "
|
||||||
#(u"Minimum Security", u"http://www.gocomics.com/minimumsecurity"),
|
# ("Ed Stein", "http://comics.com/ed_stein"),
|
||||||
#(u"Moderately Confused", u"http://www.gocomics.com/moderatelyconfused"),
|
# "Winner of the Fischetti Award in 2006 and the Scripps Howard National Journalism Award, 1999, Ed Stein has been the editorial cartoonist for the Rocky Mountain News since 1978. "
|
||||||
(u"Momma", u"http://www.gocomics.com/momma"),
|
# ("Eric Allie", "http://comics.com/eric_allie"),
|
||||||
#(u"Monty", u"http://www.gocomics.com/monty"),
|
# "Eric Allie is an editorial cartoonist with the Pioneer Press and CNS News. "
|
||||||
#(u"Motley Classics", u"http://www.gocomics.com/motleyclassics"),
|
# ("Gary Markstein", "http://comics.com/gary_markstein"),
|
||||||
(u"Mutt & Jeff", u"http://www.gocomics.com/muttandjeff"),
|
# ("Gary McCoy", "http://comics.com/gary_mccoy"),
|
||||||
#(u"Mythtickle", u"http://www.gocomics.com/mythtickle"),
|
# "Gary McCoy is known for his editorial cartoons, humor and inane ramblings. He is a 2 time nominee for Best Magazine Cartoonist of the Year by the National Cartoonists Society. He resides in Belleville, IL. "
|
||||||
#(u"Nancy", u"http://www.gocomics.com/nancy"),
|
# ("Gary Varvel", "http://comics.com/gary_varvel"),
|
||||||
#(u"Natural Selection", u"http://www.gocomics.com/naturalselection"),
|
# ("Henry Payne", "http://comics.com/henry_payne"),
|
||||||
#(u"Nest Heads", u"http://www.gocomics.com/nestheads"),
|
# ("JD Crowe", "http://comics.com/jd_crowe"),
|
||||||
#(u"NEUROTICA", u"http://www.gocomics.com/neurotica"),
|
# ("Jeff Parker", "http://comics.com/jeff_parker"),
|
||||||
#(u"New Adventures of Queen Victoria", u"http://www.gocomics.com/thenewadventuresofqueenvictoria"),
|
# ("Jeff Stahler", "http://comics.com/jeff_stahler"),
|
||||||
#(u"Non Sequitur", u"http://www.gocomics.com/nonsequitur"),
|
# ("Jerry Holbert", "http://comics.com/jerry_holbert"),
|
||||||
#(u"Off The Mark", u"http://www.gocomics.com/offthemark"),
|
# ("John Cole", "http://comics.com/john_cole"),
|
||||||
#(u"On A Claire Day", u"http://www.gocomics.com/onaclaireday"),
|
# ("John Darkow", "http://comics.com/john_darkow"),
|
||||||
#(u"One Big Happy Classics", u"http://www.gocomics.com/onebighappyclassics"),
|
# "John Darkow is a contributing editorial cartoonist for the Humor Times as well as editoiral cartoonist for the Columbia Daily Tribune, Missouri"
|
||||||
#(u"One Big Happy", u"http://www.gocomics.com/onebighappy"),
|
# ("John Sherffius", "http://comics.com/john_sherffius"),
|
||||||
#(u"Out of the Gene Pool Re-Runs", u"http://www.gocomics.com/outofthegenepool"),
|
# ("Larry Wright", "http://comics.com/larry_wright"),
|
||||||
#(u"Over the Hedge", u"http://www.gocomics.com/overthehedge"),
|
# ("Lisa Benson", "http://comics.com/lisa_benson"),
|
||||||
#(u"Overboard", u"http://www.gocomics.com/overboard"),
|
# ("Marshall Ramsey", "http://comics.com/marshall_ramsey"),
|
||||||
#(u"PC and Pixel", u"http://www.gocomics.com/pcandpixel"),
|
# ("Matt Bors", "http://comics.com/matt_bors"),
|
||||||
(u"Peanuts", u"http://www.gocomics.com/peanuts"),
|
# ("Michael Ramirez", "http://comics.com/michael_ramirez"),
|
||||||
#(u"Pearls Before Swine", u"http://www.gocomics.com/pearlsbeforeswine"),
|
# ("Mike Keefe", "http://comics.com/mike_keefe"),
|
||||||
#(u"Pibgorn Sketches", u"http://www.gocomics.com/pibgornsketches"),
|
# ("Mike Luckovich", "http://comics.com/mike_luckovich"),
|
||||||
#(u"Pibgorn", u"http://www.gocomics.com/pibgorn"),
|
# ("MIke Thompson", "http://comics.com/mike_thompson"),
|
||||||
(u"Pickles", u"http://www.gocomics.com/pickles"),
|
# ("Monte Wolverton", "http://comics.com/monte_wolverton"),
|
||||||
#(u"Pinkerton", u"http://www.gocomics.com/pinkerton"),
|
# "Unique mix of perspectives"
|
||||||
#(u"Pluggers", u"http://www.gocomics.com/pluggers"),
|
# ("Mr. Fish", "http://comics.com/mr_fish"),
|
||||||
#(u"Pooch Cafe", u"http://www.gocomics.com/poochcafe"),
|
# "Side effects may include swelling"
|
||||||
#(u"PreTeena", u"http://www.gocomics.com/preteena"),
|
# ("Nate Beeler", "http://comics.com/nate_beeler"),
|
||||||
#(u"Prickly City", u"http://www.gocomics.com/pricklycity"),
|
# "Middle America meets the Beltway."
|
||||||
#(u"Rabbits Against Magic", u"http://www.gocomics.com/rabbitsagainstmagic"),
|
# ("Nick Anderson", "http://comics.com/nick_anderson"),
|
||||||
#(u"Raising Duncan Classics", u"http://www.gocomics.com/raisingduncanclassics"),
|
# ("Pat Bagley", "http://comics.com/pat_bagley"),
|
||||||
#(u"Real Life Adventures", u"http://www.gocomics.com/reallifeadventures"),
|
# "Unfair and Totally Unbalanced."
|
||||||
#(u"Reality Check", u"http://www.gocomics.com/realitycheck"),
|
# ("Paul Szep", "http://comics.com/paul_szep"),
|
||||||
#(u"Red and Rover", u"http://www.gocomics.com/redandrover"),
|
# ("RJ Matson", "http://comics.com/rj_matson"),
|
||||||
#(u"Red Meat", u"http://www.gocomics.com/redmeat"),
|
# "Power cartoons from NYC and Capitol Hill"
|
||||||
#(u"Reynolds Unwrapped", u"http://www.gocomics.com/reynoldsunwrapped"),
|
# ("Rob Rogers", "http://comics.com/rob_rogers"),
|
||||||
#(u"Rip Haywire", u"http://www.gocomics.com/riphaywire"),
|
# "Humorous slant on current events"
|
||||||
#(u"Ripley's Believe It or Not!", u"http://www.gocomics.com/ripleysbelieveitornot"),
|
# ("Robert Ariail", "http://comics.com/robert_ariail"),
|
||||||
#(u"Ronaldinho Gaucho", u"http://www.gocomics.com/ronaldinhogaucho"),
|
# "Clever and unpredictable"
|
||||||
#(u"Rose Is Rose", u"http://www.gocomics.com/roseisrose"),
|
# ("Scott Stantis", "http://comics.com/scott_stantis"),
|
||||||
#(u"Rubes", u"http://www.gocomics.com/rubes"),
|
# ("Signe Wilkinson", "http://comics.com/signe_wilkinson"),
|
||||||
#(u"Rudy Park", u"http://www.gocomics.com/rudypark"),
|
# ("Steve Benson", "http://comics.com/steve_benson"),
|
||||||
#(u"Scary Gary", u"http://www.gocomics.com/scarygary"),
|
# ("Steve Breen", "http://comics.com/steve_breen"),
|
||||||
#(u"Shirley and Son Classics", u"http://www.gocomics.com/shirleyandsonclassics"),
|
# ("Steve Kelley", "http://comics.com/steve_kelley"),
|
||||||
#(u"Shoe", u"http://www.gocomics.com/shoe"),
|
# ("Steve Sack", "http://comics.com/steve_sack"),
|
||||||
#(u"Shoecabbage", u"http://www.gocomics.com/shoecabbage"),
|
]:
|
||||||
#(u"Skin Horse", u"http://www.gocomics.com/skinhorse"),
|
|
||||||
#(u"Slowpoke", u"http://www.gocomics.com/slowpoke"),
|
|
||||||
#(u"Soup To Nutz", u"http://www.gocomics.com/souptonutz"),
|
|
||||||
#(u"Speed Bump", u"http://www.gocomics.com/speedbump"),
|
|
||||||
#(u"Spot The Frog", u"http://www.gocomics.com/spotthefrog"),
|
|
||||||
#(u"State of the Union", u"http://www.gocomics.com/stateoftheunion"),
|
|
||||||
#(u"Stone Soup", u"http://www.gocomics.com/stonesoup"),
|
|
||||||
#(u"Strange Brew", u"http://www.gocomics.com/strangebrew"),
|
|
||||||
#(u"Sylvia", u"http://www.gocomics.com/sylvia"),
|
|
||||||
#(u"Tank McNamara", u"http://www.gocomics.com/tankmcnamara"),
|
|
||||||
#(u"Tarzan Classics", u"http://www.gocomics.com/tarzanclassics"),
|
|
||||||
#(u"That's Life", u"http://www.gocomics.com/thatslife"),
|
|
||||||
#(u"The Academia Waltz", u"http://www.gocomics.com/academiawaltz"),
|
|
||||||
#(u"The Argyle Sweater", u"http://www.gocomics.com/theargylesweater"),
|
|
||||||
#(u"The Barn", u"http://www.gocomics.com/thebarn"),
|
|
||||||
#(u"The Boiling Point", u"http://www.gocomics.com/theboilingpoint"),
|
|
||||||
#(u"The Boondocks", u"http://www.gocomics.com/boondocks"),
|
|
||||||
#(u"The Born Loser", u"http://www.gocomics.com/thebornloser"),
|
|
||||||
#(u"The Buckets", u"http://www.gocomics.com/thebuckets"),
|
|
||||||
#(u"The City", u"http://www.gocomics.com/thecity"),
|
|
||||||
#(u"The Dinette Set", u"http://www.gocomics.com/dinetteset"),
|
|
||||||
#(u"The Doozies", u"http://www.gocomics.com/thedoozies"),
|
|
||||||
#(u"The Duplex", u"http://www.gocomics.com/duplex"),
|
|
||||||
#(u"The Elderberries", u"http://www.gocomics.com/theelderberries"),
|
|
||||||
#(u"The Flying McCoys", u"http://www.gocomics.com/theflyingmccoys"),
|
|
||||||
#(u"The Fusco Brothers", u"http://www.gocomics.com/thefuscobrothers"),
|
|
||||||
#(u"The Grizzwells", u"http://www.gocomics.com/thegrizzwells"),
|
|
||||||
#(u"The Humble Stumble", u"http://www.gocomics.com/thehumblestumble"),
|
|
||||||
#(u"The Knight Life", u"http://www.gocomics.com/theknightlife"),
|
|
||||||
#(u"The Meaning of Lila", u"http://www.gocomics.com/meaningoflila"),
|
|
||||||
#(u"The Middletons", u"http://www.gocomics.com/themiddletons"),
|
|
||||||
#(u"The Norm", u"http://www.gocomics.com/thenorm"),
|
|
||||||
#(u"The Other Coast", u"http://www.gocomics.com/theothercoast"),
|
|
||||||
#(u"The Quigmans", u"http://www.gocomics.com/thequigmans"),
|
|
||||||
#(u"The Sunshine Club", u"http://www.gocomics.com/thesunshineclub"),
|
|
||||||
#(u"Tiny Sepuk", u"http://www.gocomics.com/tinysepuk"),
|
|
||||||
#(u"TOBY", u"http://www.gocomics.com/toby"),
|
|
||||||
#(u"Tom the Dancing Bug", u"http://www.gocomics.com/tomthedancingbug"),
|
|
||||||
#(u"Too Much Coffee Man", u"http://www.gocomics.com/toomuchcoffeeman"),
|
|
||||||
#(u"Unstrange Phenomena", u"http://www.gocomics.com/unstrangephenomena"),
|
|
||||||
#(u"W.T. Duck", u"http://www.gocomics.com/wtduck"),
|
|
||||||
#(u"Watch Your Head", u"http://www.gocomics.com/watchyourhead"),
|
|
||||||
#(u"Wee Pals", u"http://www.gocomics.com/weepals"),
|
|
||||||
#(u"Winnie the Pooh", u"http://www.gocomics.com/winniethepooh"),
|
|
||||||
#(u"Wizard of Id", u"http://www.gocomics.com/wizardofid"),
|
|
||||||
#(u"Working Daze", u"http://www.gocomics.com/workingdaze"),
|
|
||||||
#(u"Working It Out", u"http://www.gocomics.com/workingitout"),
|
|
||||||
#(u"Yenny", u"http://www.gocomics.com/yenny"),
|
|
||||||
#(u"Zack Hill", u"http://www.gocomics.com/zackhill"),
|
|
||||||
(u"Ziggy", u"http://www.gocomics.com/ziggy"),
|
|
||||||
#
|
|
||||||
######## EDITORIAL CARTOONS #####################
|
|
||||||
(u"Adam Zyglis", u"http://www.gocomics.com/adamzyglis"),
|
|
||||||
#(u"Andy Singer", u"http://www.gocomics.com/andysinger"),
|
|
||||||
#(u"Ben Sargent",u"http://www.gocomics.com/bensargent"),
|
|
||||||
#(u"Bill Day", u"http://www.gocomics.com/billday"),
|
|
||||||
#(u"Bill Schorr", u"http://www.gocomics.com/billschorr"),
|
|
||||||
#(u"Bob Englehart", u"http://www.gocomics.com/bobenglehart"),
|
|
||||||
(u"Bob Gorrell",u"http://www.gocomics.com/bobgorrell"),
|
|
||||||
#(u"Brian Fairrington", u"http://www.gocomics.com/brianfairrington"),
|
|
||||||
#(u"Bruce Beattie", u"http://www.gocomics.com/brucebeattie"),
|
|
||||||
#(u"Cam Cardow", u"http://www.gocomics.com/camcardow"),
|
|
||||||
#(u"Chan Lowe",u"http://www.gocomics.com/chanlowe"),
|
|
||||||
#(u"Chip Bok",u"http://www.gocomics.com/chipbok"),
|
|
||||||
#(u"Chris Britt",u"http://www.gocomics.com/chrisbritt"),
|
|
||||||
#(u"Chuck Asay",u"http://www.gocomics.com/chuckasay"),
|
|
||||||
#(u"Clay Bennett",u"http://www.gocomics.com/claybennett"),
|
|
||||||
#(u"Clay Jones",u"http://www.gocomics.com/clayjones"),
|
|
||||||
#(u"Dan Wasserman",u"http://www.gocomics.com/danwasserman"),
|
|
||||||
#(u"Dana Summers",u"http://www.gocomics.com/danasummers"),
|
|
||||||
#(u"Daryl Cagle", u"http://www.gocomics.com/darylcagle"),
|
|
||||||
#(u"David Fitzsimmons", u"http://www.gocomics.com/davidfitzsimmons"),
|
|
||||||
(u"Dick Locher",u"http://www.gocomics.com/dicklocher"),
|
|
||||||
#(u"Don Wright",u"http://www.gocomics.com/donwright"),
|
|
||||||
#(u"Donna Barstow",u"http://www.gocomics.com/donnabarstow"),
|
|
||||||
#(u"Drew Litton", u"http://www.gocomics.com/drewlitton"),
|
|
||||||
#(u"Drew Sheneman",u"http://www.gocomics.com/drewsheneman"),
|
|
||||||
#(u"Ed Stein", u"http://www.gocomics.com/edstein"),
|
|
||||||
#(u"Eric Allie", u"http://www.gocomics.com/ericallie"),
|
|
||||||
#(u"Gary Markstein", u"http://www.gocomics.com/garymarkstein"),
|
|
||||||
#(u"Gary McCoy", u"http://www.gocomics.com/garymccoy"),
|
|
||||||
#(u"Gary Varvel", u"http://www.gocomics.com/garyvarvel"),
|
|
||||||
#(u"Glenn McCoy",u"http://www.gocomics.com/glennmccoy"),
|
|
||||||
#(u"Henry Payne", u"http://www.gocomics.com/henrypayne"),
|
|
||||||
#(u"Jack Ohman",u"http://www.gocomics.com/jackohman"),
|
|
||||||
#(u"JD Crowe", u"http://www.gocomics.com/jdcrowe"),
|
|
||||||
#(u"Jeff Danziger",u"http://www.gocomics.com/jeffdanziger"),
|
|
||||||
#(u"Jeff Parker", u"http://www.gocomics.com/jeffparker"),
|
|
||||||
#(u"Jeff Stahler", u"http://www.gocomics.com/jeffstahler"),
|
|
||||||
#(u"Jerry Holbert", u"http://www.gocomics.com/jerryholbert"),
|
|
||||||
#(u"Jim Morin",u"http://www.gocomics.com/jimmorin"),
|
|
||||||
#(u"Joel Pett",u"http://www.gocomics.com/joelpett"),
|
|
||||||
#(u"John Cole", u"http://www.gocomics.com/johncole"),
|
|
||||||
#(u"John Darkow", u"http://www.gocomics.com/johndarkow"),
|
|
||||||
#(u"John Deering",u"http://www.gocomics.com/johndeering"),
|
|
||||||
#(u"John Sherffius", u"http://www.gocomics.com/johnsherffius"),
|
|
||||||
#(u"Ken Catalino",u"http://www.gocomics.com/kencatalino"),
|
|
||||||
#(u"Kerry Waghorn",u"http://www.gocomics.com/facesinthenews"),
|
|
||||||
#(u"Kevin Kallaugher",u"http://www.gocomics.com/kevinkallaugher"),
|
|
||||||
#(u"Lalo Alcaraz",u"http://www.gocomics.com/laloalcaraz"),
|
|
||||||
#(u"Larry Wright", u"http://www.gocomics.com/larrywright"),
|
|
||||||
#(u"Lisa Benson", u"http://www.gocomics.com/lisabenson"),
|
|
||||||
#(u"Marshall Ramsey", u"http://www.gocomics.com/marshallramsey"),
|
|
||||||
#(u"Matt Bors", u"http://www.gocomics.com/mattbors"),
|
|
||||||
#(u"Matt Davies",u"http://www.gocomics.com/mattdavies"),
|
|
||||||
#(u"Michael Ramirez", u"http://www.gocomics.com/michaelramirez"),
|
|
||||||
#(u"Mike Keefe", u"http://www.gocomics.com/mikekeefe"),
|
|
||||||
#(u"Mike Luckovich", u"http://www.gocomics.com/mikeluckovich"),
|
|
||||||
#(u"MIke Thompson", u"http://www.gocomics.com/mikethompson"),
|
|
||||||
#(u"Monte Wolverton", u"http://www.gocomics.com/montewolverton"),
|
|
||||||
#(u"Mr. Fish", u"http://www.gocomics.com/mrfish"),
|
|
||||||
#(u"Nate Beeler", u"http://www.gocomics.com/natebeeler"),
|
|
||||||
#(u"Nick Anderson", u"http://www.gocomics.com/nickanderson"),
|
|
||||||
#(u"Pat Bagley", u"http://www.gocomics.com/patbagley"),
|
|
||||||
#(u"Pat Oliphant",u"http://www.gocomics.com/patoliphant"),
|
|
||||||
#(u"Paul Conrad",u"http://www.gocomics.com/paulconrad"),
|
|
||||||
#(u"Paul Szep", u"http://www.gocomics.com/paulszep"),
|
|
||||||
#(u"RJ Matson", u"http://www.gocomics.com/rjmatson"),
|
|
||||||
#(u"Rob Rogers", u"http://www.gocomics.com/robrogers"),
|
|
||||||
#(u"Robert Ariail", u"http://www.gocomics.com/robertariail"),
|
|
||||||
#(u"Scott Stantis", u"http://www.gocomics.com/scottstantis"),
|
|
||||||
#(u"Signe Wilkinson", u"http://www.gocomics.com/signewilkinson"),
|
|
||||||
#(u"Small World",u"http://www.gocomics.com/smallworld"),
|
|
||||||
#(u"Steve Benson", u"http://www.gocomics.com/stevebenson"),
|
|
||||||
#(u"Steve Breen", u"http://www.gocomics.com/stevebreen"),
|
|
||||||
#(u"Steve Kelley", u"http://www.gocomics.com/stevekelley"),
|
|
||||||
#(u"Steve Sack", u"http://www.gocomics.com/stevesack"),
|
|
||||||
#(u"Stuart Carlson",u"http://www.gocomics.com/stuartcarlson"),
|
|
||||||
#(u"Ted Rall",u"http://www.gocomics.com/tedrall"),
|
|
||||||
#(u"(Th)ink", u"http://www.gocomics.com/think"),
|
|
||||||
#(u"Tom Toles",u"http://www.gocomics.com/tomtoles"),
|
|
||||||
(u"Tony Auth",u"http://www.gocomics.com/tonyauth"),
|
|
||||||
#(u"Views of the World",u"http://www.gocomics.com/viewsoftheworld"),
|
|
||||||
#(u"ViewsAfrica",u"http://www.gocomics.com/viewsafrica"),
|
|
||||||
#(u"ViewsAmerica",u"http://www.gocomics.com/viewsamerica"),
|
|
||||||
#(u"ViewsAsia",u"http://www.gocomics.com/viewsasia"),
|
|
||||||
#(u"ViewsBusiness",u"http://www.gocomics.com/viewsbusiness"),
|
|
||||||
#(u"ViewsEurope",u"http://www.gocomics.com/viewseurope"),
|
|
||||||
#(u"ViewsLatinAmerica",u"http://www.gocomics.com/viewslatinamerica"),
|
|
||||||
#(u"ViewsMidEast",u"http://www.gocomics.com/viewsmideast"),
|
|
||||||
(u"Walt Handelsman",u"http://www.gocomics.com/walthandelsman"),
|
|
||||||
#(u"Wayne Stayskal",u"http://www.gocomics.com/waynestayskal"),
|
|
||||||
#(u"Wit of the World",u"http://www.gocomics.com/witoftheworld"),
|
|
||||||
]:
|
|
||||||
print 'Working on: ', title
|
|
||||||
articles = self.make_links(url)
|
articles = self.make_links(url)
|
||||||
if articles:
|
if articles:
|
||||||
feeds.append((title, articles))
|
feeds.append((title, articles))
|
||||||
return feeds
|
return feeds
|
||||||
|
|
||||||
def make_links(self, url):
|
def make_links(self, url):
|
||||||
title = 'Temp'
|
soup = self.index_to_soup(url)
|
||||||
|
# print 'soup: ', soup
|
||||||
|
title = ''
|
||||||
current_articles = []
|
current_articles = []
|
||||||
pages = range(1, self.num_comics_to_get+1)
|
from datetime import datetime, timedelta
|
||||||
for page in pages:
|
now = datetime.now()
|
||||||
page_soup = self.index_to_soup(url)
|
dates = [(now-timedelta(days=d)).strftime('%Y/%m/%d') for d in range(self.num_comics_to_get)]
|
||||||
if page_soup:
|
|
||||||
try:
|
for page in dates:
|
||||||
strip_title = page_soup.find(name='div', attrs={'class':'top'}).h1.a.string
|
page_url = url + '/' + str(page)
|
||||||
except:
|
print(page_url)
|
||||||
strip_title = 'Error - no Title found'
|
soup = self.index_to_soup(page_url)
|
||||||
try:
|
if soup:
|
||||||
date_title = page_soup.find('ul', attrs={'class': 'feature-nav'}).li.string
|
strip_tag = self.tag_to_string(soup.find('a'))
|
||||||
if not date_title:
|
if strip_tag:
|
||||||
date_title = page_soup.find('ul', attrs={'class': 'feature-nav'}).li.string
|
print 'strip_tag: ', strip_tag
|
||||||
except:
|
title = strip_tag
|
||||||
date_title = 'Error - no Date found'
|
print 'title: ', title
|
||||||
title = strip_title + ' - ' + date_title
|
|
||||||
for i in range(2):
|
|
||||||
try:
|
|
||||||
strip_url_date = page_soup.find(name='div', attrs={'class':'top'}).h1.a['href']
|
|
||||||
break #success - this is normal exit
|
|
||||||
except:
|
|
||||||
strip_url_date = None
|
|
||||||
continue #try to get strip_url_date again
|
|
||||||
for i in range(2):
|
|
||||||
try:
|
|
||||||
prev_strip_url_date = page_soup.find('a', attrs={'class': 'prev'})['href']
|
|
||||||
break #success - this is normal exit
|
|
||||||
except:
|
|
||||||
prev_strip_url_date = None
|
|
||||||
continue #try to get prev_strip_url_date again
|
|
||||||
if strip_url_date:
|
|
||||||
page_url = 'http://www.gocomics.com' + strip_url_date
|
|
||||||
else:
|
|
||||||
continue
|
|
||||||
if prev_strip_url_date:
|
|
||||||
prev_page_url = 'http://www.gocomics.com' + prev_strip_url_date
|
|
||||||
else:
|
|
||||||
continue
|
|
||||||
current_articles.append({'title': title, 'url': page_url, 'description':'', 'date':''})
|
current_articles.append({'title': title, 'url': page_url, 'description':'', 'date':''})
|
||||||
url = prev_page_url
|
|
||||||
current_articles.reverse()
|
current_articles.reverse()
|
||||||
return current_articles
|
return current_articles
|
||||||
|
|
||||||
def preprocess_html(self, soup):
|
|
||||||
if soup.title:
|
|
||||||
title_string = soup.title.string.strip()
|
|
||||||
_cd = title_string.split(',',1)[1]
|
|
||||||
comic_date = ' '.join(_cd.split(' ', 4)[0:-1])
|
|
||||||
if soup.h1.span:
|
|
||||||
artist = soup.h1.span.string
|
|
||||||
soup.h1.span.string.replaceWith(comic_date + artist)
|
|
||||||
feature_item = soup.find('p',attrs={'class':'feature_item'})
|
|
||||||
if feature_item.a:
|
|
||||||
a_tag = feature_item.a
|
|
||||||
a_href = a_tag["href"]
|
|
||||||
img_tag = a_tag.img
|
|
||||||
img_tag["src"] = a_href
|
|
||||||
img_tag["width"] = self.comic_size
|
|
||||||
img_tag["height"] = None
|
|
||||||
return self.adeify_images(soup)
|
|
||||||
|
|
||||||
extra_css = '''
|
extra_css = '''
|
||||||
h1{font-family:Arial,Helvetica,sans-serif; font-weight:bold;font-size:large;}
|
h1{font-family:Arial,Helvetica,sans-serif; font-weight:bold;font-size:large;}
|
||||||
h2{font-family:Arial,Helvetica,sans-serif; font-weight:normal;font-size:small;}
|
h2{font-family:Arial,Helvetica,sans-serif; font-weight:normal;font-size:small;}
|
||||||
img {max-width:100%; min-width:100%;}
|
|
||||||
p{font-family:Arial,Helvetica,sans-serif;font-size:small;}
|
p{font-family:Arial,Helvetica,sans-serif;font-size:small;}
|
||||||
body{font-family:Helvetica,Arial,sans-serif;font-size:small;}
|
body{font-family:Helvetica,Arial,sans-serif;font-size:small;}
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
@ -41,6 +41,7 @@ class TheIndependentNew(BasicNewsRecipe):
|
|||||||
publication_type = 'newspaper'
|
publication_type = 'newspaper'
|
||||||
masthead_url = 'http://www.independent.co.uk/independent.co.uk/editorial/logo/independent_Masthead.png'
|
masthead_url = 'http://www.independent.co.uk/independent.co.uk/editorial/logo/independent_Masthead.png'
|
||||||
encoding = 'utf-8'
|
encoding = 'utf-8'
|
||||||
|
compress_news_images = True
|
||||||
remove_tags =[
|
remove_tags =[
|
||||||
dict(attrs={'id' : ['RelatedArtTag','renderBiography']}),
|
dict(attrs={'id' : ['RelatedArtTag','renderBiography']}),
|
||||||
dict(attrs={'class' : ['autoplay','openBiogPopup']}),
|
dict(attrs={'class' : ['autoplay','openBiogPopup']}),
|
||||||
@ -343,7 +344,7 @@ class TheIndependentNew(BasicNewsRecipe):
|
|||||||
if 'class' in subtitle_div:
|
if 'class' in subtitle_div:
|
||||||
clazz = subtitle_div['class'] + ' '
|
clazz = subtitle_div['class'] + ' '
|
||||||
clazz = clazz + 'subtitle'
|
clazz = clazz + 'subtitle'
|
||||||
subtitle_div['class'] = clazz
|
subtitle_div['class'] = clazz
|
||||||
|
|
||||||
#find broken images and remove captions
|
#find broken images and remove captions
|
||||||
items_to_extract = []
|
items_to_extract = []
|
||||||
|
11
recipes/lightspeed_magazine.recipe
Normal file
11
recipes/lightspeed_magazine.recipe
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
from calibre.web.feeds.news import BasicNewsRecipe
|
||||||
|
|
||||||
|
class AdvancedUserRecipe1366025923(BasicNewsRecipe):
|
||||||
|
title = u'Lightspeed Magazine'
|
||||||
|
language = 'en'
|
||||||
|
__author__ = 'Jose Pinto'
|
||||||
|
oldest_article = 7
|
||||||
|
max_articles_per_feed = 100
|
||||||
|
auto_cleanup = True
|
||||||
|
use_embedded_content = False
|
||||||
|
feeds = [(u'Lastest Stories', u'http://www.lightspeedmagazine.com/rss-2/')]
|
@ -11,7 +11,8 @@ class PsychologyToday(BasicNewsRecipe):
|
|||||||
language = 'en'
|
language = 'en'
|
||||||
category = 'news'
|
category = 'news'
|
||||||
encoding = 'UTF-8'
|
encoding = 'UTF-8'
|
||||||
keep_only_tags = [dict(attrs={'class':['print-title', 'print-submitted', 'print-content', 'print-footer', 'print-source_url', 'print-links']})]
|
auto_cleanup = True
|
||||||
|
#keep_only_tags = [dict(attrs={'class':['print-title', 'print-submitted', 'print-content', 'print-footer', 'print-source_url', 'print-links']})]
|
||||||
no_javascript = True
|
no_javascript = True
|
||||||
no_stylesheets = True
|
no_stylesheets = True
|
||||||
|
|
||||||
@ -31,50 +32,32 @@ class PsychologyToday(BasicNewsRecipe):
|
|||||||
self.timefmt = u' [%s]'%date
|
self.timefmt = u' [%s]'%date
|
||||||
|
|
||||||
articles = []
|
articles = []
|
||||||
for post in div.findAll('div', attrs={'class':'collections-node-feature-info'}):
|
for post in div.findAll('div', attrs={'class':'collections-node-feature collection-node-even'}):
|
||||||
title = self.tag_to_string(post.find('h2'))
|
title = self.tag_to_string(post.find('h2'))
|
||||||
author_item=post.find('div', attrs={'class':'collection-node-byline'})
|
author_item=post.find('div', attrs={'class':'collection-node-byline'})
|
||||||
author = re.sub(r'.*by\s',"",self.tag_to_string(author_item).strip())
|
author = re.sub(r'.*by\s',"",self.tag_to_string(author_item).strip())
|
||||||
title = title + u' (%s)'%author
|
title = title + u' (%s)'%author
|
||||||
article_page= self.index_to_soup('http://www.psychologytoday.com'+post.find('a', href=True)['href'])
|
url= 'http://www.psychologytoday.com'+post.find('a', href=True)['href']
|
||||||
print_page=article_page.find('li', attrs={'class':'print_html first'})
|
#print_page=article_page.find('li', attrs={'class':'print_html first'})
|
||||||
url='http://www.psychologytoday.com'+print_page.find('a',href=True)['href']
|
#url='http://www.psychologytoday.com'+print_page.find('a',href=True)['href']
|
||||||
|
desc = self.tag_to_string(post.find('div', attrs={'class':'collection-node-description'})).strip()
|
||||||
|
self.log('Found article:', title)
|
||||||
|
self.log('\t', url)
|
||||||
|
self.log('\t', desc)
|
||||||
|
articles.append({'title':title, 'url':url, 'date':'','description':desc})
|
||||||
|
for post in div.findAll('div', attrs={'class':'collections-node-feature collection-node-odd'}):
|
||||||
|
title = self.tag_to_string(post.find('h2'))
|
||||||
|
author_item=post.find('div', attrs={'class':'collection-node-byline'})
|
||||||
|
author = re.sub(r'.*by\s',"",self.tag_to_string(author_item).strip())
|
||||||
|
title = title + u' (%s)'%author
|
||||||
|
url= 'http://www.psychologytoday.com'+post.find('a', href=True)['href']
|
||||||
|
#print_page=article_page.find('li', attrs={'class':'print_html first'})
|
||||||
|
#url='http://www.psychologytoday.com'+print_page.find('a',href=True)['href']
|
||||||
desc = self.tag_to_string(post.find('div', attrs={'class':'collection-node-description'})).strip()
|
desc = self.tag_to_string(post.find('div', attrs={'class':'collection-node-description'})).strip()
|
||||||
self.log('Found article:', title)
|
self.log('Found article:', title)
|
||||||
self.log('\t', url)
|
self.log('\t', url)
|
||||||
self.log('\t', desc)
|
self.log('\t', desc)
|
||||||
articles.append({'title':title, 'url':url, 'date':'','description':desc})
|
articles.append({'title':title, 'url':url, 'date':'','description':desc})
|
||||||
|
|
||||||
for post in div.findAll('div', attrs={'class':'collections-node-thumbnail-info'}):
|
|
||||||
title = self.tag_to_string(post.find('h2'))
|
|
||||||
author_item=post.find('div', attrs={'class':'collection-node-byline'})
|
|
||||||
article_page= self.index_to_soup('http://www.psychologytoday.com'+post.find('a', href=True)['href'])
|
|
||||||
print_page=article_page.find('li', attrs={'class':'print_html first'})
|
|
||||||
description = post.find('div', attrs={'class':'collection-node-description'})
|
|
||||||
author = re.sub(r'.*by\s',"",self.tag_to_string(description.nextSibling).strip())
|
|
||||||
desc = self.tag_to_string(description).strip()
|
|
||||||
url='http://www.psychologytoday.com'+print_page.find('a',href=True)['href']
|
|
||||||
title = title + u' (%s)'%author
|
|
||||||
self.log('Found article:', title)
|
|
||||||
self.log('\t', url)
|
|
||||||
self.log('\t', desc)
|
|
||||||
articles.append({'title':title, 'url':url, 'date':'','description':desc})
|
|
||||||
|
|
||||||
for post in div.findAll('li', attrs={'class':['collection-item-list-odd','collection-item-list-even']}):
|
|
||||||
title = self.tag_to_string(post.find('h2'))
|
|
||||||
author_item=post.find('div', attrs={'class':'collection-node-byline'})
|
|
||||||
author = re.sub(r'.*by\s',"",self.tag_to_string(author_item).strip())
|
|
||||||
title = title + u' (%s)'%author
|
|
||||||
article_page= self.index_to_soup('http://www.psychologytoday.com'+post.find('a', href=True)['href'])
|
|
||||||
print_page=article_page.find('li', attrs={'class':'print_html first'})
|
|
||||||
if print_page is not None:
|
|
||||||
url='http://www.psychologytoday.com'+print_page.find('a',href=True)['href']
|
|
||||||
desc = self.tag_to_string(post.find('div', attrs={'class':'collection-node-description'})).strip()
|
|
||||||
self.log('Found article:', title)
|
|
||||||
self.log('\t', url)
|
|
||||||
self.log('\t', desc)
|
|
||||||
articles.append({'title':title, 'url':url, 'date':'','description':desc})
|
|
||||||
|
|
||||||
return [('Current Issue', articles)]
|
return [('Current Issue', articles)]
|
||||||
|
|
||||||
|
|
||||||
|
@ -50,6 +50,10 @@ class ScienceNewsIssue(BasicNewsRecipe):
|
|||||||
dict(name='ul', attrs={'id':'toc'})
|
dict(name='ul', attrs={'id':'toc'})
|
||||||
]
|
]
|
||||||
|
|
||||||
|
remove_tags= [ dict(name='a', attrs={'class':'enlarge print-no'}),
|
||||||
|
dict(name='a', attrs={'rel':'shadowbox'})
|
||||||
|
]
|
||||||
|
|
||||||
feeds = [(u"Science News Current Issues", u'http://www.sciencenews.org/view/feed/type/edition/name/issues.rss')]
|
feeds = [(u"Science News Current Issues", u'http://www.sciencenews.org/view/feed/type/edition/name/issues.rss')]
|
||||||
|
|
||||||
match_regexps = [
|
match_regexps = [
|
||||||
@ -57,6 +61,12 @@ class ScienceNewsIssue(BasicNewsRecipe):
|
|||||||
r'www.sciencenews.org/view/generic/id'
|
r'www.sciencenews.org/view/generic/id'
|
||||||
]
|
]
|
||||||
|
|
||||||
|
def image_url_processor(self, baseurl, url):
|
||||||
|
x = url.split('/')
|
||||||
|
if x[4] == u'scale':
|
||||||
|
url = u'http://www.sciencenews.org/view/download/id/' + x[6] + u'/name/' + x[-1]
|
||||||
|
return url
|
||||||
|
|
||||||
def get_cover_url(self):
|
def get_cover_url(self):
|
||||||
cover_url = None
|
cover_url = None
|
||||||
index = 'http://www.sciencenews.org/view/home'
|
index = 'http://www.sciencenews.org/view/home'
|
||||||
@ -64,7 +74,6 @@ class ScienceNewsIssue(BasicNewsRecipe):
|
|||||||
link_item = soup.find(name = 'img',alt = "issue")
|
link_item = soup.find(name = 'img',alt = "issue")
|
||||||
if link_item:
|
if link_item:
|
||||||
cover_url = 'http://www.sciencenews.org' + link_item['src'] + '.jpg'
|
cover_url = 'http://www.sciencenews.org' + link_item['src'] + '.jpg'
|
||||||
|
|
||||||
return cover_url
|
return cover_url
|
||||||
|
|
||||||
def preprocess_html(self, soup):
|
def preprocess_html(self, soup):
|
||||||
|
@ -1,8 +1,11 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
__license__ = 'GPL v3'
|
__license__ = 'GPL v3'
|
||||||
__author__ = 'Lorenzo Vigentini'
|
__author__ = 'Lorenzo Vigentini and Tom Surace'
|
||||||
__copyright__ = '2009, Lorenzo Vigentini <l.vigentini at gmail.com>'
|
__copyright__ = '2009, Lorenzo Vigentini <l.vigentini at gmail.com>, 2013 Tom Surace <tekhedd@byteheaven.net>'
|
||||||
description = 'the Escapist Magazine - v1.02 (09, January 2010)'
|
description = 'The Escapist Magazine - v1.3 (2013, April 2013)'
|
||||||
|
|
||||||
|
#
|
||||||
|
# Based on 'the Escapist Magazine - v1.02 (09, January 2010)'
|
||||||
|
|
||||||
'''
|
'''
|
||||||
http://www.escapistmagazine.com/
|
http://www.escapistmagazine.com/
|
||||||
@ -11,12 +14,11 @@ http://www.escapistmagazine.com/
|
|||||||
from calibre.web.feeds.news import BasicNewsRecipe
|
from calibre.web.feeds.news import BasicNewsRecipe
|
||||||
|
|
||||||
class al(BasicNewsRecipe):
|
class al(BasicNewsRecipe):
|
||||||
author = 'Lorenzo Vigentini'
|
author = 'Lorenzo Vigentini and Tom Surace'
|
||||||
description = 'The Escapist Magazine'
|
description = 'The Escapist Magazine'
|
||||||
|
|
||||||
cover_url = 'http://cdn.themis-media.com/themes/escapistmagazine/default/images/logo.png'
|
cover_url = 'http://cdn.themis-media.com/themes/escapistmagazine/default/images/logo.png'
|
||||||
title = u'The Escapist Magazine'
|
title = u'The Escapist Magazine'
|
||||||
publisher = 'Themis media'
|
publisher = 'Themis Media'
|
||||||
category = 'Video games news, lifestyle, gaming culture'
|
category = 'Video games news, lifestyle, gaming culture'
|
||||||
|
|
||||||
language = 'en'
|
language = 'en'
|
||||||
@ -36,18 +38,19 @@ class al(BasicNewsRecipe):
|
|||||||
]
|
]
|
||||||
|
|
||||||
def print_version(self,url):
|
def print_version(self,url):
|
||||||
|
# Expect article url in the format:
|
||||||
|
# http://www.escapistmagazine.com/news/view/123198-article-name?utm_source=rss&utm_medium=rss&utm_campaign=news
|
||||||
|
#
|
||||||
baseURL='http://www.escapistmagazine.com'
|
baseURL='http://www.escapistmagazine.com'
|
||||||
segments = url.split('/')
|
segments = url.split('/')
|
||||||
#basename = '/'.join(segments[:3]) + '/'
|
|
||||||
subPath= '/'+ segments[3] + '/'
|
subPath= '/'+ segments[3] + '/'
|
||||||
articleURL=(segments[len(segments)-1])[0:5]
|
|
||||||
|
|
||||||
if articleURL[4] =='-':
|
# The article number is the "number" that starts the name
|
||||||
articleURL=articleURL[:4]
|
articleNumber = segments[len(segments)-1]; # the "article name"
|
||||||
|
articleNumber = articleNumber.split('-')[0]; # keep part before hyphen
|
||||||
|
|
||||||
printVerString='print/'+ articleURL
|
fullUrl = baseURL + subPath + 'print/' + articleNumber
|
||||||
s= baseURL + subPath + printVerString
|
return fullUrl
|
||||||
return s
|
|
||||||
|
|
||||||
keep_only_tags = [
|
keep_only_tags = [
|
||||||
dict(name='div', attrs={'id':'article'})
|
dict(name='div', attrs={'id':'article'})
|
||||||
|
11
recipes/the_feature.recipe
Normal file
11
recipes/the_feature.recipe
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
from calibre.web.feeds.news import BasicNewsRecipe
|
||||||
|
|
||||||
|
class AdvancedUserRecipe1365777047(BasicNewsRecipe):
|
||||||
|
title = u'The Feature'
|
||||||
|
__author__ = 'Jose Pinto'
|
||||||
|
language = 'en'
|
||||||
|
oldest_article = 30
|
||||||
|
max_articles_per_feed = 100
|
||||||
|
auto_cleanup = True
|
||||||
|
use_embedded_content = False
|
||||||
|
feeds = [(u'Latest', u'http://thefeature.net/rss/links')]
|
@ -1,5 +1,6 @@
|
|||||||
# vim:fileencoding=UTF-8:ts=4:sw=4:sta:et:sts=4:ai
|
# vim:fileencoding=UTF-8:ts=4:sw=4:sta:et:sts=4:ai
|
||||||
|
|
||||||
|
import re
|
||||||
from calibre.web.feeds.recipes import BasicNewsRecipe
|
from calibre.web.feeds.recipes import BasicNewsRecipe
|
||||||
|
|
||||||
class TVXS(BasicNewsRecipe):
|
class TVXS(BasicNewsRecipe):
|
||||||
@ -8,19 +9,30 @@ class TVXS(BasicNewsRecipe):
|
|||||||
description = 'News from Greece'
|
description = 'News from Greece'
|
||||||
max_articles_per_feed = 100
|
max_articles_per_feed = 100
|
||||||
oldest_article = 3
|
oldest_article = 3
|
||||||
simultaneous_downloads = 1
|
|
||||||
publisher = 'TVXS'
|
publisher = 'TVXS'
|
||||||
category = 'news, GR'
|
category = 'news, sport, greece'
|
||||||
language = 'el'
|
language = 'el'
|
||||||
encoding = None
|
encoding = None
|
||||||
use_embedded_content = False
|
use_embedded_content = False
|
||||||
remove_empty_feeds = True
|
remove_empty_feeds = True
|
||||||
#conversion_options = { 'linearize_tables': True}
|
conversion_options = {'smarten_punctuation': True}
|
||||||
no_stylesheets = True
|
no_stylesheets = True
|
||||||
|
publication_type = 'newspaper'
|
||||||
remove_tags_before = dict(name='h1',attrs={'class':'print-title'})
|
remove_tags_before = dict(name='h1',attrs={'class':'print-title'})
|
||||||
remove_tags_after = dict(name='div',attrs={'class':'field field-type-relevant-content field-field-relevant-articles'})
|
remove_tags_after = dict(name='div',attrs={'class':'field field-type-relevant-content field-field-relevant-articles'})
|
||||||
remove_attributes = ['width', 'src', 'header', 'footer']
|
remove_tags = [dict(name='div',attrs={'class':'field field-type-relevant-content field-field-relevant-articles'}),
|
||||||
|
dict(name='div',attrs={'class':'field field-type-filefield field-field-image-gallery'}),
|
||||||
|
dict(name='div',attrs={'class':'filefield-file'})]
|
||||||
|
remove_attributes = ['border', 'cellspacing', 'align', 'cellpadding', 'colspan', 'valign', 'vspace', 'hspace', 'alt', 'width', 'height']
|
||||||
|
extra_css = 'body { font-family: verdana, helvetica, sans-serif; } \
|
||||||
|
table { width: 100%; } \
|
||||||
|
td img { display: block; margin: 5px auto; } \
|
||||||
|
ul { padding-top: 10px; } \
|
||||||
|
ol { padding-top: 10px; } \
|
||||||
|
li { padding-top: 5px; padding-bottom: 5px; } \
|
||||||
|
h1 { text-align: center; font-size: 125%; font-weight: bold; } \
|
||||||
|
h2, h3, h4, h5, h6 { text-align: center; font-size: 100%; font-weight: bold; }'
|
||||||
|
preprocess_regexps = [(re.compile(r'<br[ ]*/>', re.IGNORECASE), lambda m: ''), (re.compile(r'<br[ ]*clear.*/>', re.IGNORECASE), lambda m: '')]
|
||||||
|
|
||||||
feeds = [(u'Ελλάδα', 'http://tvxs.gr/feeds/2/feed.xml'),
|
feeds = [(u'Ελλάδα', 'http://tvxs.gr/feeds/2/feed.xml'),
|
||||||
(u'Κόσμος', 'http://tvxs.gr/feeds/5/feed.xml'),
|
(u'Κόσμος', 'http://tvxs.gr/feeds/5/feed.xml'),
|
||||||
@ -35,17 +47,10 @@ class TVXS(BasicNewsRecipe):
|
|||||||
(u'Ιστορία', 'http://tvxs.gr/feeds/1573/feed.xml'),
|
(u'Ιστορία', 'http://tvxs.gr/feeds/1573/feed.xml'),
|
||||||
(u'Χιούμορ', 'http://tvxs.gr/feeds/692/feed.xml')]
|
(u'Χιούμορ', 'http://tvxs.gr/feeds/692/feed.xml')]
|
||||||
|
|
||||||
|
|
||||||
def print_version(self, url):
|
def print_version(self, url):
|
||||||
import urllib2, urlparse, StringIO, gzip
|
br = self.get_browser()
|
||||||
|
response = br.open(url)
|
||||||
fp = urllib2.urlopen(url)
|
data = response.read()
|
||||||
data = fp.read()
|
|
||||||
if fp.info()['content-encoding'] == 'gzip':
|
|
||||||
gzip_data = StringIO.StringIO(data)
|
|
||||||
gzipper = gzip.GzipFile(fileobj=gzip_data)
|
|
||||||
data = gzipper.read()
|
|
||||||
fp.close()
|
|
||||||
|
|
||||||
pos_1 = data.find('<a href="/print/')
|
pos_1 = data.find('<a href="/print/')
|
||||||
if pos_1 == -1:
|
if pos_1 == -1:
|
||||||
@ -57,5 +62,5 @@ class TVXS(BasicNewsRecipe):
|
|||||||
pos_1 += len('<a href="')
|
pos_1 += len('<a href="')
|
||||||
new_url = data[pos_1:pos_2]
|
new_url = data[pos_1:pos_2]
|
||||||
|
|
||||||
print_url = urlparse.urljoin(url, new_url)
|
print_url = "http://tvxs.gr" + new_url
|
||||||
return print_url
|
return print_url
|
||||||
|
@ -448,8 +448,15 @@
|
|||||||
<xsl:template match = "rtf:field[@type='hyperlink']">
|
<xsl:template match = "rtf:field[@type='hyperlink']">
|
||||||
<xsl:element name ="a">
|
<xsl:element name ="a">
|
||||||
<xsl:attribute name = "href">
|
<xsl:attribute name = "href">
|
||||||
<xsl:if test = "not(contains(@link, '/'))">#</xsl:if>
|
<xsl:choose>
|
||||||
<xsl:value-of select = "@link"/>
|
<xsl:when test="@argument">
|
||||||
|
<xsl:value-of select="@argument"/>
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<xsl:if test = "not(contains(@link, '/'))">#</xsl:if>
|
||||||
|
<xsl:value-of select = "@link"/>
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
</xsl:attribute>
|
</xsl:attribute>
|
||||||
<xsl:apply-templates/>
|
<xsl:apply-templates/>
|
||||||
</xsl:element>
|
</xsl:element>
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -4,7 +4,7 @@ __license__ = 'GPL v3'
|
|||||||
__copyright__ = '2008, Kovid Goyal kovid@kovidgoyal.net'
|
__copyright__ = '2008, Kovid Goyal kovid@kovidgoyal.net'
|
||||||
__docformat__ = 'restructuredtext en'
|
__docformat__ = 'restructuredtext en'
|
||||||
__appname__ = u'calibre'
|
__appname__ = u'calibre'
|
||||||
numeric_version = (0, 9, 26)
|
numeric_version = (0, 9, 27)
|
||||||
__version__ = u'.'.join(map(unicode, numeric_version))
|
__version__ = u'.'.join(map(unicode, numeric_version))
|
||||||
__author__ = u"Kovid Goyal <kovid@kovidgoyal.net>"
|
__author__ = u"Kovid Goyal <kovid@kovidgoyal.net>"
|
||||||
|
|
||||||
|
@ -1468,6 +1468,17 @@ class StoreKoboStore(StoreBase):
|
|||||||
formats = ['EPUB']
|
formats = ['EPUB']
|
||||||
affiliate = True
|
affiliate = True
|
||||||
|
|
||||||
|
class StoreKoobeStore(StoreBase):
|
||||||
|
name = 'Koobe'
|
||||||
|
author = u'Tomasz Długosz'
|
||||||
|
description = u'Księgarnia internetowa oferuje ebooki (książki elektroniczne) w postaci plików epub, mobi i pdf.'
|
||||||
|
actual_plugin = 'calibre.gui2.store.stores.koobe_plugin:KoobeStore'
|
||||||
|
|
||||||
|
drm_free_only = True
|
||||||
|
headquarters = 'PL'
|
||||||
|
formats = ['EPUB', 'MOBI', 'PDF']
|
||||||
|
affiliate = True
|
||||||
|
|
||||||
class StoreLegimiStore(StoreBase):
|
class StoreLegimiStore(StoreBase):
|
||||||
name = 'Legimi'
|
name = 'Legimi'
|
||||||
author = u'Tomasz Długosz'
|
author = u'Tomasz Długosz'
|
||||||
@ -1650,6 +1661,7 @@ class StoreWoblinkStore(StoreBase):
|
|||||||
|
|
||||||
headquarters = 'PL'
|
headquarters = 'PL'
|
||||||
formats = ['EPUB', 'MOBI', 'PDF', 'WOBLINK']
|
formats = ['EPUB', 'MOBI', 'PDF', 'WOBLINK']
|
||||||
|
affiliate = True
|
||||||
|
|
||||||
class XinXiiStore(StoreBase):
|
class XinXiiStore(StoreBase):
|
||||||
name = 'XinXii'
|
name = 'XinXii'
|
||||||
@ -1687,6 +1699,7 @@ plugins += [
|
|||||||
StoreGoogleBooksStore,
|
StoreGoogleBooksStore,
|
||||||
StoreGutenbergStore,
|
StoreGutenbergStore,
|
||||||
StoreKoboStore,
|
StoreKoboStore,
|
||||||
|
StoreKoobeStore,
|
||||||
StoreLegimiStore,
|
StoreLegimiStore,
|
||||||
StoreLibreDEStore,
|
StoreLibreDEStore,
|
||||||
StoreLitResStore,
|
StoreLitResStore,
|
||||||
|
@ -306,7 +306,8 @@ class DB(object):
|
|||||||
|
|
||||||
# Initialize database {{{
|
# Initialize database {{{
|
||||||
|
|
||||||
def __init__(self, library_path, default_prefs=None, read_only=False):
|
def __init__(self, library_path, default_prefs=None, read_only=False,
|
||||||
|
restore_all_prefs=False, progress_callback=lambda x, y:True):
|
||||||
try:
|
try:
|
||||||
if isbytestring(library_path):
|
if isbytestring(library_path):
|
||||||
library_path = library_path.decode(filesystem_encoding)
|
library_path = library_path.decode(filesystem_encoding)
|
||||||
@ -377,23 +378,27 @@ class DB(object):
|
|||||||
UPDATE authors SET sort=author_to_author_sort(name) WHERE sort IS NULL;
|
UPDATE authors SET sort=author_to_author_sort(name) WHERE sort IS NULL;
|
||||||
''')
|
''')
|
||||||
|
|
||||||
self.initialize_prefs(default_prefs)
|
self.initialize_prefs(default_prefs, restore_all_prefs, progress_callback)
|
||||||
self.initialize_custom_columns()
|
self.initialize_custom_columns()
|
||||||
self.initialize_tables()
|
self.initialize_tables()
|
||||||
|
|
||||||
def initialize_prefs(self, default_prefs): # {{{
|
def initialize_prefs(self, default_prefs, restore_all_prefs, progress_callback): # {{{
|
||||||
self.prefs = DBPrefs(self)
|
self.prefs = DBPrefs(self)
|
||||||
|
|
||||||
if default_prefs is not None and not self._exists:
|
if default_prefs is not None and not self._exists:
|
||||||
|
progress_callback(None, len(default_prefs))
|
||||||
# Only apply default prefs to a new database
|
# Only apply default prefs to a new database
|
||||||
for key in default_prefs:
|
for i, key in enumerate(default_prefs):
|
||||||
# be sure that prefs not to be copied are listed below
|
# be sure that prefs not to be copied are listed below
|
||||||
if key not in frozenset(['news_to_be_synced']):
|
if restore_all_prefs or key not in frozenset(['news_to_be_synced']):
|
||||||
self.prefs[key] = default_prefs[key]
|
self.prefs[key] = default_prefs[key]
|
||||||
|
progress_callback(_('restored preference ') + key, i+1)
|
||||||
if 'field_metadata' in default_prefs:
|
if 'field_metadata' in default_prefs:
|
||||||
fmvals = [f for f in default_prefs['field_metadata'].values()
|
fmvals = [f for f in default_prefs['field_metadata'].values()
|
||||||
if f['is_custom']]
|
if f['is_custom']]
|
||||||
for f in fmvals:
|
progress_callback(None, len(fmvals))
|
||||||
|
for i, f in enumerate(fmvals):
|
||||||
|
progress_callback(_('creating custom column ') + f['label'], i)
|
||||||
self.create_custom_column(f['label'], f['name'],
|
self.create_custom_column(f['label'], f['name'],
|
||||||
f['datatype'],
|
f['datatype'],
|
||||||
(f['is_multiple'] is not None and
|
(f['is_multiple'] is not None and
|
||||||
@ -422,6 +427,8 @@ class DB(object):
|
|||||||
('uuid', False), ('comments', True), ('id', False), ('pubdate', False),
|
('uuid', False), ('comments', True), ('id', False), ('pubdate', False),
|
||||||
('last_modified', False), ('size', False), ('languages', False),
|
('last_modified', False), ('size', False), ('languages', False),
|
||||||
]
|
]
|
||||||
|
defs['virtual_libraries'] = {}
|
||||||
|
defs['virtual_lib_on_startup'] = defs['cs_virtual_lib_on_startup'] = ''
|
||||||
|
|
||||||
# Migrate the bool tristate tweak
|
# Migrate the bool tristate tweak
|
||||||
defs['bools_are_tristate'] = \
|
defs['bools_are_tristate'] = \
|
||||||
@ -470,6 +477,24 @@ class DB(object):
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
# migrate the gui_restriction preference to a virtual library
|
||||||
|
gr_pref = self.prefs.get('gui_restriction', None)
|
||||||
|
if gr_pref:
|
||||||
|
virt_libs = self.prefs.get('virtual_libraries', {})
|
||||||
|
virt_libs[gr_pref] = 'search:"' + gr_pref + '"'
|
||||||
|
self.prefs['virtual_libraries'] = virt_libs
|
||||||
|
self.prefs['gui_restriction'] = ''
|
||||||
|
self.prefs['virtual_lib_on_startup'] = gr_pref
|
||||||
|
|
||||||
|
# migrate the cs_restriction preference to a virtual library
|
||||||
|
gr_pref = self.prefs.get('cs_restriction', None)
|
||||||
|
if gr_pref:
|
||||||
|
virt_libs = self.prefs.get('virtual_libraries', {})
|
||||||
|
virt_libs[gr_pref] = 'search:"' + gr_pref + '"'
|
||||||
|
self.prefs['virtual_libraries'] = virt_libs
|
||||||
|
self.prefs['cs_restriction'] = ''
|
||||||
|
self.prefs['cs_virtual_lib_on_startup'] = gr_pref
|
||||||
|
|
||||||
# Rename any user categories with names that differ only in case
|
# Rename any user categories with names that differ only in case
|
||||||
user_cats = self.prefs.get('user_categories', [])
|
user_cats = self.prefs.get('user_categories', [])
|
||||||
catmap = {}
|
catmap = {}
|
||||||
@ -691,11 +716,13 @@ class DB(object):
|
|||||||
|
|
||||||
tables['size'] = SizeTable('size', self.field_metadata['size'].copy())
|
tables['size'] = SizeTable('size', self.field_metadata['size'].copy())
|
||||||
|
|
||||||
self.FIELD_MAP = {'id':0, 'title':1, 'authors':2, 'timestamp':3,
|
self.FIELD_MAP = {
|
||||||
'size':4, 'rating':5, 'tags':6, 'comments':7, 'series':8,
|
'id':0, 'title':1, 'authors':2, 'timestamp':3, 'size':4,
|
||||||
'publisher':9, 'series_index':10, 'sort':11, 'author_sort':12,
|
'rating':5, 'tags':6, 'comments':7, 'series':8, 'publisher':9,
|
||||||
'formats':13, 'path':14, 'pubdate':15, 'uuid':16, 'cover':17,
|
'series_index':10, 'sort':11, 'author_sort':12, 'formats':13,
|
||||||
'au_map':18, 'last_modified':19, 'identifiers':20}
|
'path':14, 'pubdate':15, 'uuid':16, 'cover':17, 'au_map':18,
|
||||||
|
'last_modified':19, 'identifiers':20, 'languages':21,
|
||||||
|
}
|
||||||
|
|
||||||
for k,v in self.FIELD_MAP.iteritems():
|
for k,v in self.FIELD_MAP.iteritems():
|
||||||
self.field_metadata.set_field_record_index(k, v, prefer_custom=False)
|
self.field_metadata.set_field_record_index(k, v, prefer_custom=False)
|
||||||
@ -741,6 +768,8 @@ class DB(object):
|
|||||||
self.field_metadata.set_field_record_index('ondevice', base, prefer_custom=False)
|
self.field_metadata.set_field_record_index('ondevice', base, prefer_custom=False)
|
||||||
self.FIELD_MAP['marked'] = base = base+1
|
self.FIELD_MAP['marked'] = base = base+1
|
||||||
self.field_metadata.set_field_record_index('marked', base, prefer_custom=False)
|
self.field_metadata.set_field_record_index('marked', base, prefer_custom=False)
|
||||||
|
self.FIELD_MAP['series_sort'] = base = base+1
|
||||||
|
self.field_metadata.set_field_record_index('series_sort', base, prefer_custom=False)
|
||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
@ -754,6 +783,11 @@ class DB(object):
|
|||||||
self._conn = Connection(self.dbpath)
|
self._conn = Connection(self.dbpath)
|
||||||
return self._conn
|
return self._conn
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
if self._conn is not None:
|
||||||
|
self._conn.close()
|
||||||
|
del self._conn
|
||||||
|
|
||||||
@dynamic_property
|
@dynamic_property
|
||||||
def user_version(self):
|
def user_version(self):
|
||||||
doc = 'The user version of this database'
|
doc = 'The user version of this database'
|
||||||
|
94
src/calibre/db/legacy.py
Normal file
94
src/calibre/db/legacy.py
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# vim:fileencoding=utf-8
|
||||||
|
from __future__ import (unicode_literals, division, absolute_import,
|
||||||
|
print_function)
|
||||||
|
|
||||||
|
__license__ = 'GPL v3'
|
||||||
|
__copyright__ = '2013, Kovid Goyal <kovid at kovidgoyal.net>'
|
||||||
|
|
||||||
|
import os
|
||||||
|
from functools import partial
|
||||||
|
|
||||||
|
from calibre.db.backend import DB
|
||||||
|
from calibre.db.cache import Cache
|
||||||
|
from calibre.db.view import View
|
||||||
|
|
||||||
|
class LibraryDatabase(object):
|
||||||
|
|
||||||
|
''' Emulate the old LibraryDatabase2 interface '''
|
||||||
|
|
||||||
|
PATH_LIMIT = DB.PATH_LIMIT
|
||||||
|
WINDOWS_LIBRARY_PATH_LIMIT = DB.WINDOWS_LIBRARY_PATH_LIMIT
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def exists_at(cls, path):
|
||||||
|
return path and os.path.exists(os.path.join(path, 'metadata.db'))
|
||||||
|
|
||||||
|
def __init__(self, library_path,
|
||||||
|
default_prefs=None, read_only=False, is_second_db=False,
|
||||||
|
progress_callback=lambda x, y:True, restore_all_prefs=False):
|
||||||
|
|
||||||
|
self.is_second_db = is_second_db # TODO: Use is_second_db
|
||||||
|
|
||||||
|
backend = self.backend = DB(library_path, default_prefs=default_prefs,
|
||||||
|
read_only=read_only, restore_all_prefs=restore_all_prefs,
|
||||||
|
progress_callback=progress_callback)
|
||||||
|
cache = self.new_api = Cache(backend)
|
||||||
|
cache.init()
|
||||||
|
self.data = View(cache)
|
||||||
|
|
||||||
|
self.get_property = self.data.get_property
|
||||||
|
|
||||||
|
for prop in (
|
||||||
|
'author_sort', 'authors', 'comment', 'comments',
|
||||||
|
'publisher', 'rating', 'series', 'series_index', 'tags',
|
||||||
|
'title', 'timestamp', 'uuid', 'pubdate', 'ondevice',
|
||||||
|
'metadata_last_modified', 'languages',
|
||||||
|
):
|
||||||
|
fm = {'comment':'comments', 'metadata_last_modified':
|
||||||
|
'last_modified', 'title_sort':'sort'}.get(prop, prop)
|
||||||
|
setattr(self, prop, partial(self.get_property,
|
||||||
|
loc=self.FIELD_MAP[fm]))
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
self.backend.close()
|
||||||
|
|
||||||
|
def break_cycles(self):
|
||||||
|
self.data.cache.backend = None
|
||||||
|
self.data.cache = None
|
||||||
|
self.data = self.backend = self.new_api = self.field_metadata = self.prefs = self.listeners = self.refresh_ondevice = None
|
||||||
|
|
||||||
|
# Library wide properties {{{
|
||||||
|
@property
|
||||||
|
def field_metadata(self):
|
||||||
|
return self.backend.field_metadata
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user_version(self):
|
||||||
|
return self.backend.user_version
|
||||||
|
|
||||||
|
@property
|
||||||
|
def library_id(self):
|
||||||
|
return self.backend.library_id
|
||||||
|
|
||||||
|
def last_modified(self):
|
||||||
|
return self.backend.last_modified()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def custom_column_num_map(self):
|
||||||
|
return self.backend.custom_column_num_map
|
||||||
|
|
||||||
|
@property
|
||||||
|
def custom_column_label_map(self):
|
||||||
|
return self.backend.custom_column_label_map
|
||||||
|
|
||||||
|
@property
|
||||||
|
def FIELD_MAP(self):
|
||||||
|
return self.backend.FIELD_MAP
|
||||||
|
|
||||||
|
def all_ids(self):
|
||||||
|
for book_id in self.data.cache.all_book_ids():
|
||||||
|
yield book_id
|
||||||
|
# }}}
|
||||||
|
|
||||||
|
|
@ -16,6 +16,9 @@ rmtree = partial(shutil.rmtree, ignore_errors=True)
|
|||||||
|
|
||||||
class BaseTest(unittest.TestCase):
|
class BaseTest(unittest.TestCase):
|
||||||
|
|
||||||
|
longMessage = True
|
||||||
|
maxDiff = None
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.library_path = self.mkdtemp()
|
self.library_path = self.mkdtemp()
|
||||||
self.create_db(self.library_path)
|
self.create_db(self.library_path)
|
||||||
@ -40,10 +43,10 @@ class BaseTest(unittest.TestCase):
|
|||||||
db.conn.close()
|
db.conn.close()
|
||||||
return dest
|
return dest
|
||||||
|
|
||||||
def init_cache(self, library_path):
|
def init_cache(self, library_path=None):
|
||||||
from calibre.db.backend import DB
|
from calibre.db.backend import DB
|
||||||
from calibre.db.cache import Cache
|
from calibre.db.cache import Cache
|
||||||
backend = DB(library_path)
|
backend = DB(library_path or self.library_path)
|
||||||
cache = Cache(backend)
|
cache = Cache(backend)
|
||||||
cache.init()
|
cache.init()
|
||||||
return cache
|
return cache
|
||||||
@ -53,9 +56,13 @@ class BaseTest(unittest.TestCase):
|
|||||||
atexit.register(rmtree, ans)
|
atexit.register(rmtree, ans)
|
||||||
return ans
|
return ans
|
||||||
|
|
||||||
def init_old(self, library_path):
|
def init_old(self, library_path=None):
|
||||||
from calibre.library.database2 import LibraryDatabase2
|
from calibre.library.database2 import LibraryDatabase2
|
||||||
return LibraryDatabase2(library_path)
|
return LibraryDatabase2(library_path or self.library_path)
|
||||||
|
|
||||||
|
def init_legacy(self, library_path=None):
|
||||||
|
from calibre.db.legacy import LibraryDatabase
|
||||||
|
return LibraryDatabase(library_path or self.library_path)
|
||||||
|
|
||||||
def clone_library(self, library_path):
|
def clone_library(self, library_path):
|
||||||
if not hasattr(self, 'clone_dir'):
|
if not hasattr(self, 'clone_dir'):
|
||||||
@ -81,7 +88,8 @@ class BaseTest(unittest.TestCase):
|
|||||||
'ondevice_col', 'last_modified', 'has_cover',
|
'ondevice_col', 'last_modified', 'has_cover',
|
||||||
'cover_data'}.union(allfk1)
|
'cover_data'}.union(allfk1)
|
||||||
for attr in all_keys:
|
for attr in all_keys:
|
||||||
if attr == 'user_metadata': continue
|
if attr == 'user_metadata':
|
||||||
|
continue
|
||||||
attr1, attr2 = getattr(mi1, attr), getattr(mi2, attr)
|
attr1, attr2 = getattr(mi1, attr), getattr(mi2, attr)
|
||||||
if attr == 'formats':
|
if attr == 'formats':
|
||||||
attr1, attr2 = map(lambda x:tuple(x) if x else (), (attr1, attr2))
|
attr1, attr2 = map(lambda x:tuple(x) if x else (), (attr1, attr2))
|
||||||
|
66
src/calibre/db/tests/legacy.py
Normal file
66
src/calibre/db/tests/legacy.py
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# vim:fileencoding=utf-8
|
||||||
|
from __future__ import (unicode_literals, division, absolute_import,
|
||||||
|
print_function)
|
||||||
|
|
||||||
|
__license__ = 'GPL v3'
|
||||||
|
__copyright__ = '2013, Kovid Goyal <kovid at kovidgoyal.net>'
|
||||||
|
|
||||||
|
from calibre.db.tests.base import BaseTest
|
||||||
|
|
||||||
|
class LegacyTest(BaseTest):
|
||||||
|
|
||||||
|
''' Test the emulation of the legacy interface. '''
|
||||||
|
|
||||||
|
def test_library_wide_properties(self): # {{{
|
||||||
|
'Test library wide properties'
|
||||||
|
def get_props(db):
|
||||||
|
props = ('user_version', 'is_second_db', 'library_id', 'field_metadata',
|
||||||
|
'custom_column_label_map', 'custom_column_num_map')
|
||||||
|
fprops = ('last_modified', )
|
||||||
|
ans = {x:getattr(db, x) for x in props}
|
||||||
|
ans.update({x:getattr(db, x)() for x in fprops})
|
||||||
|
ans['all_ids'] = frozenset(db.all_ids())
|
||||||
|
return ans
|
||||||
|
|
||||||
|
old = self.init_old()
|
||||||
|
oldvals = get_props(old)
|
||||||
|
old.close()
|
||||||
|
del old
|
||||||
|
db = self.init_legacy()
|
||||||
|
newvals = get_props(db)
|
||||||
|
self.assertEqual(oldvals, newvals)
|
||||||
|
db.close()
|
||||||
|
# }}}
|
||||||
|
|
||||||
|
def test_get_property(self): # {{{
|
||||||
|
'Test the get_property interface for reading data'
|
||||||
|
def get_values(db):
|
||||||
|
ans = {}
|
||||||
|
for label, loc in db.FIELD_MAP.iteritems():
|
||||||
|
if isinstance(label, int):
|
||||||
|
label = '#'+db.custom_column_num_map[label]['label']
|
||||||
|
label = type('')(label)
|
||||||
|
ans[label] = tuple(db.get_property(i, index_is_id=True, loc=loc)
|
||||||
|
for i in db.all_ids())
|
||||||
|
if label in ('id', 'title', '#tags'):
|
||||||
|
with self.assertRaises(IndexError):
|
||||||
|
db.get_property(9999, loc=loc)
|
||||||
|
with self.assertRaises(IndexError):
|
||||||
|
db.get_property(9999, index_is_id=True, loc=loc)
|
||||||
|
if label in {'tags', 'formats'}:
|
||||||
|
# Order is random in the old db for these
|
||||||
|
ans[label] = tuple(set(x.split(',')) if x else x for x in ans[label])
|
||||||
|
return ans
|
||||||
|
|
||||||
|
old = self.init_old()
|
||||||
|
old_vals = get_values(old)
|
||||||
|
old.close()
|
||||||
|
old = None
|
||||||
|
db = self.init_legacy()
|
||||||
|
new_vals = get_values(db)
|
||||||
|
db.close()
|
||||||
|
self.assertEqual(old_vals, new_vals)
|
||||||
|
|
||||||
|
# }}}
|
||||||
|
|
@ -11,6 +11,9 @@ import weakref
|
|||||||
from functools import partial
|
from functools import partial
|
||||||
from itertools import izip, imap
|
from itertools import izip, imap
|
||||||
|
|
||||||
|
from calibre.ebooks.metadata import title_sort
|
||||||
|
from calibre.utils.config_base import tweaks
|
||||||
|
|
||||||
def sanitize_sort_field_name(field_metadata, field):
|
def sanitize_sort_field_name(field_metadata, field):
|
||||||
field = field_metadata.search_term_to_field_key(field.lower().strip())
|
field = field_metadata.search_term_to_field_key(field.lower().strip())
|
||||||
# translate some fields to their hidden equivalent
|
# translate some fields to their hidden equivalent
|
||||||
@ -40,6 +43,18 @@ class TableRow(list):
|
|||||||
else:
|
else:
|
||||||
return view._field_getters[obj](self.book_id)
|
return view._field_getters[obj](self.book_id)
|
||||||
|
|
||||||
|
def format_is_multiple(x, sep=',', repl=None):
|
||||||
|
if not x:
|
||||||
|
return None
|
||||||
|
if repl is not None:
|
||||||
|
x = (y.replace(sep, repl) for y in x)
|
||||||
|
return sep.join(x)
|
||||||
|
|
||||||
|
def format_identifiers(x):
|
||||||
|
if not x:
|
||||||
|
return None
|
||||||
|
return ','.join('%s:%s'%(k, v) for k, v in x.iteritems())
|
||||||
|
|
||||||
class View(object):
|
class View(object):
|
||||||
|
|
||||||
''' A table view of the database, with rows and columns. Also supports
|
''' A table view of the database, with rows and columns. Also supports
|
||||||
@ -49,33 +64,63 @@ class View(object):
|
|||||||
self.cache = cache
|
self.cache = cache
|
||||||
self.marked_ids = {}
|
self.marked_ids = {}
|
||||||
self.search_restriction_book_count = 0
|
self.search_restriction_book_count = 0
|
||||||
self.search_restriction = ''
|
self.search_restriction = self.base_restriction = ''
|
||||||
|
self.search_restriction_name = self.base_restriction_name = ''
|
||||||
self._field_getters = {}
|
self._field_getters = {}
|
||||||
for col, idx in cache.backend.FIELD_MAP.iteritems():
|
for col, idx in cache.backend.FIELD_MAP.iteritems():
|
||||||
|
label, fmt = col, lambda x:x
|
||||||
|
func = {
|
||||||
|
'id': self._get_id,
|
||||||
|
'au_map': self.get_author_data,
|
||||||
|
'ondevice': self.get_ondevice,
|
||||||
|
'marked': self.get_marked,
|
||||||
|
'series_sort':self.get_series_sort,
|
||||||
|
}.get(col, self._get)
|
||||||
if isinstance(col, int):
|
if isinstance(col, int):
|
||||||
label = self.cache.backend.custom_column_num_map[col]['label']
|
label = self.cache.backend.custom_column_num_map[col]['label']
|
||||||
label = (self.cache.backend.field_metadata.custom_field_prefix
|
label = (self.cache.backend.field_metadata.custom_field_prefix
|
||||||
+ label)
|
+ label)
|
||||||
self._field_getters[idx] = partial(self.get, label)
|
if label.endswith('_index'):
|
||||||
else:
|
|
||||||
try:
|
try:
|
||||||
self._field_getters[idx] = {
|
num = int(label.partition('_')[0])
|
||||||
'id': self._get_id,
|
except ValueError:
|
||||||
'au_map': self.get_author_data,
|
pass # series_index
|
||||||
'ondevice': self.get_ondevice,
|
else:
|
||||||
'marked': self.get_marked,
|
label = self.cache.backend.custom_column_num_map[num]['label']
|
||||||
}[col]
|
label = (self.cache.backend.field_metadata.custom_field_prefix
|
||||||
except KeyError:
|
+ label + '_index')
|
||||||
self._field_getters[idx] = partial(self.get, col)
|
|
||||||
|
fm = self.field_metadata[label]
|
||||||
|
fm
|
||||||
|
if label == 'authors':
|
||||||
|
fmt = partial(format_is_multiple, repl='|')
|
||||||
|
elif label in {'tags', 'languages', 'formats'}:
|
||||||
|
fmt = format_is_multiple
|
||||||
|
elif label == 'cover':
|
||||||
|
fmt = bool
|
||||||
|
elif label == 'identifiers':
|
||||||
|
fmt = format_identifiers
|
||||||
|
elif fm['datatype'] == 'text' and fm['is_multiple']:
|
||||||
|
sep = fm['is_multiple']['cache_to_list']
|
||||||
|
if sep not in {'&','|'}:
|
||||||
|
sep = '|'
|
||||||
|
fmt = partial(format_is_multiple, sep=sep)
|
||||||
|
self._field_getters[idx] = partial(func, label, fmt=fmt) if func == self._get else func
|
||||||
|
|
||||||
self._map = tuple(self.cache.all_book_ids())
|
self._map = tuple(self.cache.all_book_ids())
|
||||||
self._map_filtered = tuple(self._map)
|
self._map_filtered = tuple(self._map)
|
||||||
|
|
||||||
|
def get_property(self, id_or_index, index_is_id=False, loc=-1):
|
||||||
|
book_id = id_or_index if index_is_id else self._map_filtered[id_or_index]
|
||||||
|
return self._field_getters[loc](book_id)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def field_metadata(self):
|
def field_metadata(self):
|
||||||
return self.cache.field_metadata
|
return self.cache.field_metadata
|
||||||
|
|
||||||
def _get_id(self, idx, index_is_id=True):
|
def _get_id(self, idx, index_is_id=True):
|
||||||
|
if index_is_id and idx not in self.cache.all_book_ids():
|
||||||
|
raise IndexError('No book with id %s present'%idx)
|
||||||
return idx if index_is_id else self.index_to_id(idx)
|
return idx if index_is_id else self.index_to_id(idx)
|
||||||
|
|
||||||
def __getitem__(self, row):
|
def __getitem__(self, row):
|
||||||
@ -107,9 +152,21 @@ class View(object):
|
|||||||
def index_to_id(self, idx):
|
def index_to_id(self, idx):
|
||||||
return self._map_filtered[idx]
|
return self._map_filtered[idx]
|
||||||
|
|
||||||
def get(self, field, idx, index_is_id=True, default_value=None):
|
def _get(self, field, idx, index_is_id=True, default_value=None, fmt=lambda x:x):
|
||||||
id_ = idx if index_is_id else self.index_to_id(idx)
|
id_ = idx if index_is_id else self.index_to_id(idx)
|
||||||
return self.cache.field_for(field, id_)
|
if index_is_id and id_ not in self.cache.all_book_ids():
|
||||||
|
raise IndexError('No book with id %s present'%idx)
|
||||||
|
return fmt(self.cache.field_for(field, id_, default_value=default_value))
|
||||||
|
|
||||||
|
def get_series_sort(self, idx, index_is_id=True, default_value=''):
|
||||||
|
book_id = idx if index_is_id else self.index_to_id(idx)
|
||||||
|
with self.cache.read_lock:
|
||||||
|
lang_map = self.cache.fields['languages'].book_value_map
|
||||||
|
lang = lang_map.get(book_id, None) or None
|
||||||
|
if lang:
|
||||||
|
lang = lang[0]
|
||||||
|
return title_sort(self.cache._field_for('series', book_id, default_value=''),
|
||||||
|
order=tweaks['title_series_sorting'], lang=lang)
|
||||||
|
|
||||||
def get_ondevice(self, idx, index_is_id=True, default_value=''):
|
def get_ondevice(self, idx, index_is_id=True, default_value=''):
|
||||||
id_ = idx if index_is_id else self.index_to_id(idx)
|
id_ = idx if index_is_id else self.index_to_id(idx)
|
||||||
@ -119,26 +176,15 @@ class View(object):
|
|||||||
id_ = idx if index_is_id else self.index_to_id(idx)
|
id_ = idx if index_is_id else self.index_to_id(idx)
|
||||||
return self.marked_ids.get(id_, default_value)
|
return self.marked_ids.get(id_, default_value)
|
||||||
|
|
||||||
def get_author_data(self, idx, index_is_id=True, default_value=()):
|
def get_author_data(self, idx, index_is_id=True, default_value=None):
|
||||||
'''
|
|
||||||
Return author data for all authors of the book identified by idx as a
|
|
||||||
tuple of dictionaries. The dictionaries should never be empty, unless
|
|
||||||
there is a bug somewhere. The list could be empty if idx point to an
|
|
||||||
non existent book, or book with no authors (though again a book with no
|
|
||||||
authors should never happen).
|
|
||||||
|
|
||||||
Each dictionary has the keys: name, sort, link. Link can be an empty
|
|
||||||
string.
|
|
||||||
|
|
||||||
default_value is ignored, this method always returns a tuple
|
|
||||||
'''
|
|
||||||
id_ = idx if index_is_id else self.index_to_id(idx)
|
id_ = idx if index_is_id else self.index_to_id(idx)
|
||||||
with self.cache.read_lock:
|
with self.cache.read_lock:
|
||||||
ids = self.cache._field_ids_for('authors', id_)
|
ids = self.cache._field_ids_for('authors', id_)
|
||||||
ans = []
|
ans = []
|
||||||
for id_ in ids:
|
for id_ in ids:
|
||||||
ans.append(self.cache._author_data(id_))
|
data = self.cache._author_data(id_)
|
||||||
return tuple(ans)
|
ans.append(':::'.join((data['name'], data['sort'], data['link'])))
|
||||||
|
return ':#:'.join(ans) if ans else default_value
|
||||||
|
|
||||||
def multisort(self, fields=[], subsort=False, only_ids=None):
|
def multisort(self, fields=[], subsort=False, only_ids=None):
|
||||||
fields = [(sanitize_sort_field_name(self.field_metadata, x), bool(y)) for x, y in fields]
|
fields = [(sanitize_sort_field_name(self.field_metadata, x), bool(y)) for x, y in fields]
|
||||||
@ -168,8 +214,19 @@ class View(object):
|
|||||||
return ans
|
return ans
|
||||||
self._map_filtered = tuple(ans)
|
self._map_filtered = tuple(ans)
|
||||||
|
|
||||||
|
def _build_restriction_string(self, restriction):
|
||||||
|
if self.base_restriction:
|
||||||
|
if restriction:
|
||||||
|
return u'(%s) and (%s)' % (self.base_restriction, restriction)
|
||||||
|
else:
|
||||||
|
return self.base_restriction
|
||||||
|
else:
|
||||||
|
return restriction
|
||||||
|
|
||||||
def search_getting_ids(self, query, search_restriction,
|
def search_getting_ids(self, query, search_restriction,
|
||||||
set_restriction_count=False):
|
set_restriction_count=False, use_virtual_library=True):
|
||||||
|
if use_virtual_library:
|
||||||
|
search_restriction = self._build_restriction_string(search_restriction)
|
||||||
q = ''
|
q = ''
|
||||||
if not query or not query.strip():
|
if not query or not query.strip():
|
||||||
q = search_restriction
|
q = search_restriction
|
||||||
@ -188,11 +245,32 @@ class View(object):
|
|||||||
self.search_restriction_book_count = len(rv)
|
self.search_restriction_book_count = len(rv)
|
||||||
return rv
|
return rv
|
||||||
|
|
||||||
|
def get_search_restriction(self):
|
||||||
|
return self.search_restriction
|
||||||
|
|
||||||
def set_search_restriction(self, s):
|
def set_search_restriction(self, s):
|
||||||
self.search_restriction = s
|
self.search_restriction = s
|
||||||
|
|
||||||
|
def get_base_restriction(self):
|
||||||
|
return self.base_restriction
|
||||||
|
|
||||||
|
def set_base_restriction(self, s):
|
||||||
|
self.base_restriction = s
|
||||||
|
|
||||||
|
def get_base_restriction_name(self):
|
||||||
|
return self.base_restriction_name
|
||||||
|
|
||||||
|
def set_base_restriction_name(self, s):
|
||||||
|
self.base_restriction_name = s
|
||||||
|
|
||||||
|
def get_search_restriction_name(self):
|
||||||
|
return self.search_restriction_name
|
||||||
|
|
||||||
|
def set_search_restriction_name(self, s):
|
||||||
|
self.search_restriction_name = s
|
||||||
|
|
||||||
def search_restriction_applied(self):
|
def search_restriction_applied(self):
|
||||||
return bool(self.search_restriction)
|
return bool(self.search_restriction) or bool(self.base_restriction)
|
||||||
|
|
||||||
def get_search_restriction_book_count(self):
|
def get_search_restriction_book_count(self):
|
||||||
return self.search_restriction_book_count
|
return self.search_restriction_book_count
|
||||||
|
@ -24,7 +24,7 @@ from calibre import prints, guess_type
|
|||||||
from calibre.utils.cleantext import clean_ascii_chars
|
from calibre.utils.cleantext import clean_ascii_chars
|
||||||
from calibre.utils.config import tweaks
|
from calibre.utils.config import tweaks
|
||||||
|
|
||||||
class Resource(object): # {{{
|
class Resource(object): # {{{
|
||||||
'''
|
'''
|
||||||
Represents a resource (usually a file on the filesystem or a URL pointing
|
Represents a resource (usually a file on the filesystem or a URL pointing
|
||||||
to the web. Such resources are commonly referred to in OPF files.
|
to the web. Such resources are commonly referred to in OPF files.
|
||||||
@ -68,7 +68,6 @@ class Resource(object): # {{{
|
|||||||
self.path = os.path.abspath(os.path.join(basedir, pc.replace('/', os.sep)))
|
self.path = os.path.abspath(os.path.join(basedir, pc.replace('/', os.sep)))
|
||||||
self.fragment = url[-1]
|
self.fragment = url[-1]
|
||||||
|
|
||||||
|
|
||||||
def href(self, basedir=None):
|
def href(self, basedir=None):
|
||||||
'''
|
'''
|
||||||
Return a URL pointing to this resource. If it is a file on the filesystem
|
Return a URL pointing to this resource. If it is a file on the filesystem
|
||||||
@ -90,7 +89,7 @@ class Resource(object): # {{{
|
|||||||
return ''+frag
|
return ''+frag
|
||||||
try:
|
try:
|
||||||
rpath = os.path.relpath(self.path, basedir)
|
rpath = os.path.relpath(self.path, basedir)
|
||||||
except ValueError: # On windows path and basedir could be on different drives
|
except ValueError: # On windows path and basedir could be on different drives
|
||||||
rpath = self.path
|
rpath = self.path
|
||||||
if isinstance(rpath, unicode):
|
if isinstance(rpath, unicode):
|
||||||
rpath = rpath.encode('utf-8')
|
rpath = rpath.encode('utf-8')
|
||||||
@ -107,7 +106,7 @@ class Resource(object): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class ResourceCollection(object): # {{{
|
class ResourceCollection(object): # {{{
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self._resources = []
|
self._resources = []
|
||||||
@ -160,7 +159,7 @@ class ResourceCollection(object): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class ManifestItem(Resource): # {{{
|
class ManifestItem(Resource): # {{{
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_opf_manifest_item(item, basedir):
|
def from_opf_manifest_item(item, basedir):
|
||||||
@ -180,7 +179,6 @@ class ManifestItem(Resource): # {{{
|
|||||||
self.mime_type = val
|
self.mime_type = val
|
||||||
return property(fget=fget, fset=fset)
|
return property(fget=fget, fset=fset)
|
||||||
|
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return u'<item id="%s" href="%s" media-type="%s" />'%(self.id, self.href(), self.media_type)
|
return u'<item id="%s" href="%s" media-type="%s" />'%(self.id, self.href(), self.media_type)
|
||||||
|
|
||||||
@ -190,7 +188,6 @@ class ManifestItem(Resource): # {{{
|
|||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return unicode(self)
|
return unicode(self)
|
||||||
|
|
||||||
|
|
||||||
def __getitem__(self, index):
|
def __getitem__(self, index):
|
||||||
if index == 0:
|
if index == 0:
|
||||||
return self.href()
|
return self.href()
|
||||||
@ -200,7 +197,7 @@ class ManifestItem(Resource): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class Manifest(ResourceCollection): # {{{
|
class Manifest(ResourceCollection): # {{{
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_opf_manifest_element(items, dir):
|
def from_opf_manifest_element(items, dir):
|
||||||
@ -245,7 +242,6 @@ class Manifest(ResourceCollection): # {{{
|
|||||||
ResourceCollection.__init__(self)
|
ResourceCollection.__init__(self)
|
||||||
self.next_id = 1
|
self.next_id = 1
|
||||||
|
|
||||||
|
|
||||||
def item(self, id):
|
def item(self, id):
|
||||||
for i in self:
|
for i in self:
|
||||||
if i.id == id:
|
if i.id == id:
|
||||||
@ -269,7 +265,7 @@ class Manifest(ResourceCollection): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class Spine(ResourceCollection): # {{{
|
class Spine(ResourceCollection): # {{{
|
||||||
|
|
||||||
class Item(Resource):
|
class Item(Resource):
|
||||||
|
|
||||||
@ -309,13 +305,10 @@ class Spine(ResourceCollection): # {{{
|
|||||||
continue
|
continue
|
||||||
return s
|
return s
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def __init__(self, manifest):
|
def __init__(self, manifest):
|
||||||
ResourceCollection.__init__(self)
|
ResourceCollection.__init__(self)
|
||||||
self.manifest = manifest
|
self.manifest = manifest
|
||||||
|
|
||||||
|
|
||||||
def replace(self, start, end, ids):
|
def replace(self, start, end, ids):
|
||||||
'''
|
'''
|
||||||
Replace the items between start (inclusive) and end (not inclusive) with
|
Replace the items between start (inclusive) and end (not inclusive) with
|
||||||
@ -345,7 +338,7 @@ class Spine(ResourceCollection): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class Guide(ResourceCollection): # {{{
|
class Guide(ResourceCollection): # {{{
|
||||||
|
|
||||||
class Reference(Resource):
|
class Reference(Resource):
|
||||||
|
|
||||||
@ -363,7 +356,6 @@ class Guide(ResourceCollection): # {{{
|
|||||||
ans += 'title="%s" '%self.title
|
ans += 'title="%s" '%self.title
|
||||||
return ans + '/>'
|
return ans + '/>'
|
||||||
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_opf_guide(references, base_dir=os.getcwdu()):
|
def from_opf_guide(references, base_dir=os.getcwdu()):
|
||||||
coll = Guide()
|
coll = Guide()
|
||||||
@ -484,14 +476,14 @@ def dump_dict(cats):
|
|||||||
return json.dumps(object_to_unicode(cats), ensure_ascii=False,
|
return json.dumps(object_to_unicode(cats), ensure_ascii=False,
|
||||||
skipkeys=True)
|
skipkeys=True)
|
||||||
|
|
||||||
class OPF(object): # {{{
|
class OPF(object): # {{{
|
||||||
|
|
||||||
MIMETYPE = 'application/oebps-package+xml'
|
MIMETYPE = 'application/oebps-package+xml'
|
||||||
PARSER = etree.XMLParser(recover=True)
|
PARSER = etree.XMLParser(recover=True)
|
||||||
NAMESPACES = {
|
NAMESPACES = {
|
||||||
None : "http://www.idpf.org/2007/opf",
|
None: "http://www.idpf.org/2007/opf",
|
||||||
'dc' : "http://purl.org/dc/elements/1.1/",
|
'dc': "http://purl.org/dc/elements/1.1/",
|
||||||
'opf' : "http://www.idpf.org/2007/opf",
|
'opf': "http://www.idpf.org/2007/opf",
|
||||||
}
|
}
|
||||||
META = '{%s}meta' % NAMESPACES['opf']
|
META = '{%s}meta' % NAMESPACES['opf']
|
||||||
xpn = NAMESPACES.copy()
|
xpn = NAMESPACES.copy()
|
||||||
@ -501,9 +493,10 @@ class OPF(object): # {{{
|
|||||||
CONTENT = XPath('self::*[re:match(name(), "meta$", "i")]/@content')
|
CONTENT = XPath('self::*[re:match(name(), "meta$", "i")]/@content')
|
||||||
TEXT = XPath('string()')
|
TEXT = XPath('string()')
|
||||||
|
|
||||||
|
|
||||||
metadata_path = XPath('descendant::*[re:match(name(), "metadata", "i")]')
|
metadata_path = XPath('descendant::*[re:match(name(), "metadata", "i")]')
|
||||||
metadata_elem_path = XPath('descendant::*[re:match(name(), concat($name, "$"), "i") or (re:match(name(), "meta$", "i") and re:match(@name, concat("^calibre:", $name, "$"), "i"))]')
|
metadata_elem_path = XPath(
|
||||||
|
'descendant::*[re:match(name(), concat($name, "$"), "i") or (re:match(name(), "meta$", "i") '
|
||||||
|
'and re:match(@name, concat("^calibre:", $name, "$"), "i"))]')
|
||||||
title_path = XPath('descendant::*[re:match(name(), "title", "i")]')
|
title_path = XPath('descendant::*[re:match(name(), "title", "i")]')
|
||||||
authors_path = XPath('descendant::*[re:match(name(), "creator", "i") and (@role="aut" or @opf:role="aut" or (not(@role) and not(@opf:role)))]')
|
authors_path = XPath('descendant::*[re:match(name(), "creator", "i") and (@role="aut" or @opf:role="aut" or (not(@role) and not(@opf:role)))]')
|
||||||
bkp_path = XPath('descendant::*[re:match(name(), "contributor", "i") and (@role="bkp" or @opf:role="bkp")]')
|
bkp_path = XPath('descendant::*[re:match(name(), "contributor", "i") and (@role="bkp" or @opf:role="bkp")]')
|
||||||
@ -640,7 +633,8 @@ class OPF(object): # {{{
|
|||||||
if 'toc' in item.href().lower():
|
if 'toc' in item.href().lower():
|
||||||
toc = item.path
|
toc = item.path
|
||||||
|
|
||||||
if toc is None: return
|
if toc is None:
|
||||||
|
return
|
||||||
self.toc = TOC(base_path=self.base_dir)
|
self.toc = TOC(base_path=self.base_dir)
|
||||||
is_ncx = getattr(self, 'manifest', None) is not None and \
|
is_ncx = getattr(self, 'manifest', None) is not None and \
|
||||||
self.manifest.type_for_id(toc) is not None and \
|
self.manifest.type_for_id(toc) is not None and \
|
||||||
@ -976,7 +970,6 @@ class OPF(object): # {{{
|
|||||||
|
|
||||||
return property(fget=fget, fset=fset)
|
return property(fget=fget, fset=fset)
|
||||||
|
|
||||||
|
|
||||||
@dynamic_property
|
@dynamic_property
|
||||||
def language(self):
|
def language(self):
|
||||||
|
|
||||||
@ -990,7 +983,6 @@ class OPF(object): # {{{
|
|||||||
|
|
||||||
return property(fget=fget, fset=fset)
|
return property(fget=fget, fset=fset)
|
||||||
|
|
||||||
|
|
||||||
@dynamic_property
|
@dynamic_property
|
||||||
def languages(self):
|
def languages(self):
|
||||||
|
|
||||||
@ -1015,7 +1007,6 @@ class OPF(object): # {{{
|
|||||||
|
|
||||||
return property(fget=fget, fset=fset)
|
return property(fget=fget, fset=fset)
|
||||||
|
|
||||||
|
|
||||||
@dynamic_property
|
@dynamic_property
|
||||||
def book_producer(self):
|
def book_producer(self):
|
||||||
|
|
||||||
@ -1196,7 +1187,6 @@ class OPFCreator(Metadata):
|
|||||||
if self.cover:
|
if self.cover:
|
||||||
self.guide.set_cover(self.cover)
|
self.guide.set_cover(self.cover)
|
||||||
|
|
||||||
|
|
||||||
def create_manifest(self, entries):
|
def create_manifest(self, entries):
|
||||||
'''
|
'''
|
||||||
Create <manifest>
|
Create <manifest>
|
||||||
@ -1615,9 +1605,9 @@ def test_user_metadata():
|
|||||||
from cStringIO import StringIO
|
from cStringIO import StringIO
|
||||||
mi = Metadata('Test title', ['test author1', 'test author2'])
|
mi = Metadata('Test title', ['test author1', 'test author2'])
|
||||||
um = {
|
um = {
|
||||||
'#myseries': { '#value#': u'test series\xe4', 'datatype':'text',
|
'#myseries': {'#value#': u'test series\xe4', 'datatype':'text',
|
||||||
'is_multiple': None, 'name': u'My Series'},
|
'is_multiple': None, 'name': u'My Series'},
|
||||||
'#myseries_index': { '#value#': 2.45, 'datatype': 'float',
|
'#myseries_index': {'#value#': 2.45, 'datatype': 'float',
|
||||||
'is_multiple': None},
|
'is_multiple': None},
|
||||||
'#mytags': {'#value#':['t1','t2','t3'], 'datatype':'text',
|
'#mytags': {'#value#':['t1','t2','t3'], 'datatype':'text',
|
||||||
'is_multiple': '|', 'name': u'My Tags'}
|
'is_multiple': '|', 'name': u'My Tags'}
|
||||||
|
@ -51,9 +51,11 @@ def reverse_tag_iter(block):
|
|||||||
end = len(block)
|
end = len(block)
|
||||||
while True:
|
while True:
|
||||||
pgt = block.rfind(b'>', 0, end)
|
pgt = block.rfind(b'>', 0, end)
|
||||||
if pgt == -1: break
|
if pgt == -1:
|
||||||
|
break
|
||||||
plt = block.rfind(b'<', 0, pgt)
|
plt = block.rfind(b'<', 0, pgt)
|
||||||
if plt == -1: break
|
if plt == -1:
|
||||||
|
break
|
||||||
yield block[plt:pgt+1]
|
yield block[plt:pgt+1]
|
||||||
end = plt
|
end = plt
|
||||||
|
|
||||||
@ -231,12 +233,12 @@ class Mobi8Reader(object):
|
|||||||
flowpart = self.flows[j]
|
flowpart = self.flows[j]
|
||||||
nstr = '%04d' % j
|
nstr = '%04d' % j
|
||||||
m = svg_tag_pattern.search(flowpart)
|
m = svg_tag_pattern.search(flowpart)
|
||||||
if m != None:
|
if m is not None:
|
||||||
# svg
|
# svg
|
||||||
typ = 'svg'
|
typ = 'svg'
|
||||||
start = m.start()
|
start = m.start()
|
||||||
m2 = image_tag_pattern.search(flowpart)
|
m2 = image_tag_pattern.search(flowpart)
|
||||||
if m2 != None:
|
if m2 is not None:
|
||||||
format = 'inline'
|
format = 'inline'
|
||||||
dir = None
|
dir = None
|
||||||
fname = None
|
fname = None
|
||||||
@ -320,7 +322,7 @@ class Mobi8Reader(object):
|
|||||||
if len(pos_fid) != 2:
|
if len(pos_fid) != 2:
|
||||||
continue
|
continue
|
||||||
except TypeError:
|
except TypeError:
|
||||||
continue # thumbnailstandard record, ignore it
|
continue # thumbnailstandard record, ignore it
|
||||||
linktgt, idtext = self.get_id_tag_by_pos_fid(*pos_fid)
|
linktgt, idtext = self.get_id_tag_by_pos_fid(*pos_fid)
|
||||||
if idtext:
|
if idtext:
|
||||||
linktgt += b'#' + idtext
|
linktgt += b'#' + idtext
|
||||||
@ -389,7 +391,7 @@ class Mobi8Reader(object):
|
|||||||
href = None
|
href = None
|
||||||
if typ in {b'FLIS', b'FCIS', b'SRCS', b'\xe9\x8e\r\n',
|
if typ in {b'FLIS', b'FCIS', b'SRCS', b'\xe9\x8e\r\n',
|
||||||
b'RESC', b'BOUN', b'FDST', b'DATP', b'AUDI', b'VIDE'}:
|
b'RESC', b'BOUN', b'FDST', b'DATP', b'AUDI', b'VIDE'}:
|
||||||
pass # Ignore these records
|
pass # Ignore these records
|
||||||
elif typ == b'FONT':
|
elif typ == b'FONT':
|
||||||
font = read_font_record(data)
|
font = read_font_record(data)
|
||||||
href = "fonts/%05d.%s" % (fname_idx, font['ext'])
|
href = "fonts/%05d.%s" % (fname_idx, font['ext'])
|
||||||
@ -406,7 +408,11 @@ class Mobi8Reader(object):
|
|||||||
else:
|
else:
|
||||||
imgtype = what(None, data)
|
imgtype = what(None, data)
|
||||||
if imgtype is None:
|
if imgtype is None:
|
||||||
imgtype = 'unknown'
|
from calibre.utils.magick.draw import identify_data
|
||||||
|
try:
|
||||||
|
imgtype = identify_data(data)[2]
|
||||||
|
except Exception:
|
||||||
|
imgtype = 'unknown'
|
||||||
href = 'images/%05d.%s'%(fname_idx, imgtype)
|
href = 'images/%05d.%s'%(fname_idx, imgtype)
|
||||||
with open(href.replace('/', os.sep), 'wb') as f:
|
with open(href.replace('/', os.sep), 'wb') as f:
|
||||||
f.write(data)
|
f.write(data)
|
||||||
|
@ -19,7 +19,7 @@ from calibre.ebooks.mobi.reader.mobi8 import Mobi8Reader
|
|||||||
from calibre.ebooks.conversion.plumber import Plumber, create_oebbook
|
from calibre.ebooks.conversion.plumber import Plumber, create_oebbook
|
||||||
from calibre.customize.ui import (plugin_for_input_format,
|
from calibre.customize.ui import (plugin_for_input_format,
|
||||||
plugin_for_output_format)
|
plugin_for_output_format)
|
||||||
from calibre.utils.ipc.simple_worker import fork_job
|
from calibre.utils.ipc.simple_worker import fork_job
|
||||||
|
|
||||||
class BadFormat(ValueError):
|
class BadFormat(ValueError):
|
||||||
pass
|
pass
|
||||||
@ -72,7 +72,8 @@ def explode(path, dest, question=lambda x:True):
|
|||||||
dest), no_output=True)['result']
|
dest), no_output=True)['result']
|
||||||
|
|
||||||
def set_cover(oeb):
|
def set_cover(oeb):
|
||||||
if 'cover' not in oeb.guide or oeb.metadata['cover']: return
|
if 'cover' not in oeb.guide or oeb.metadata['cover']:
|
||||||
|
return
|
||||||
cover = oeb.guide['cover']
|
cover = oeb.guide['cover']
|
||||||
if cover.href in oeb.manifest.hrefs:
|
if cover.href in oeb.manifest.hrefs:
|
||||||
item = oeb.manifest.hrefs[cover.href]
|
item = oeb.manifest.hrefs[cover.href]
|
||||||
@ -95,8 +96,9 @@ def rebuild(src_dir, dest_path):
|
|||||||
if not opf:
|
if not opf:
|
||||||
raise ValueError('No OPF file found in %s'%src_dir)
|
raise ValueError('No OPF file found in %s'%src_dir)
|
||||||
opf = opf[0]
|
opf = opf[0]
|
||||||
# For debugging, uncomment the following line
|
# For debugging, uncomment the following two lines
|
||||||
# def fork_job(a, b, args=None, no_output=True): do_rebuild(*args)
|
# def fork_job(a, b, args=None, no_output=True):
|
||||||
|
# do_rebuild(*args)
|
||||||
fork_job('calibre.ebooks.mobi.tweak', 'do_rebuild', args=(opf, dest_path),
|
fork_job('calibre.ebooks.mobi.tweak', 'do_rebuild', args=(opf, dest_path),
|
||||||
no_output=True)
|
no_output=True)
|
||||||
|
|
||||||
|
@ -69,7 +69,8 @@ class Resources(object):
|
|||||||
cover_href = item.href
|
cover_href = item.href
|
||||||
|
|
||||||
for item in self.oeb.manifest.values():
|
for item in self.oeb.manifest.values():
|
||||||
if item.media_type not in OEB_RASTER_IMAGES: continue
|
if item.media_type not in OEB_RASTER_IMAGES:
|
||||||
|
continue
|
||||||
try:
|
try:
|
||||||
data = self.process_image(item.data)
|
data = self.process_image(item.data)
|
||||||
except:
|
except:
|
||||||
@ -116,8 +117,8 @@ class Resources(object):
|
|||||||
Add any images that were created after the call to add_resources()
|
Add any images that were created after the call to add_resources()
|
||||||
'''
|
'''
|
||||||
for item in self.oeb.manifest.values():
|
for item in self.oeb.manifest.values():
|
||||||
if (item.media_type not in OEB_RASTER_IMAGES or item.href in
|
if (item.media_type not in OEB_RASTER_IMAGES or item.href in self.item_map):
|
||||||
self.item_map): continue
|
continue
|
||||||
try:
|
try:
|
||||||
data = self.process_image(item.data)
|
data = self.process_image(item.data)
|
||||||
except:
|
except:
|
||||||
|
@ -270,7 +270,7 @@ BINARY_MIME = 'application/octet-stream'
|
|||||||
|
|
||||||
XHTML_CSS_NAMESPACE = u'@namespace "%s";\n' % XHTML_NS
|
XHTML_CSS_NAMESPACE = u'@namespace "%s";\n' % XHTML_NS
|
||||||
|
|
||||||
OEB_STYLES = set([CSS_MIME, OEB_CSS_MIME, 'text/x-oeb-css'])
|
OEB_STYLES = set([CSS_MIME, OEB_CSS_MIME, 'text/x-oeb-css', 'xhtml/css'])
|
||||||
OEB_DOCS = set([XHTML_MIME, 'text/html', OEB_DOC_MIME,
|
OEB_DOCS = set([XHTML_MIME, 'text/html', OEB_DOC_MIME,
|
||||||
'text/x-oeb-document'])
|
'text/x-oeb-document'])
|
||||||
OEB_RASTER_IMAGES = set([GIF_MIME, JPEG_MIME, PNG_MIME])
|
OEB_RASTER_IMAGES = set([GIF_MIME, JPEG_MIME, PNG_MIME])
|
||||||
|
@ -30,7 +30,7 @@ from calibre.ebooks.oeb.base import (
|
|||||||
from calibre.ebooks.oeb.polish.errors import InvalidBook, DRMError
|
from calibre.ebooks.oeb.polish.errors import InvalidBook, DRMError
|
||||||
from calibre.ebooks.oeb.parse_utils import NotHTML, parse_html, RECOVER_PARSER
|
from calibre.ebooks.oeb.parse_utils import NotHTML, parse_html, RECOVER_PARSER
|
||||||
from calibre.ptempfile import PersistentTemporaryDirectory, PersistentTemporaryFile
|
from calibre.ptempfile import PersistentTemporaryDirectory, PersistentTemporaryFile
|
||||||
from calibre.utils.ipc.simple_worker import fork_job, WorkerError
|
from calibre.utils.ipc.simple_worker import fork_job, WorkerError
|
||||||
from calibre.utils.logging import default_log
|
from calibre.utils.logging import default_log
|
||||||
from calibre.utils.zipfile import ZipFile
|
from calibre.utils.zipfile import ZipFile
|
||||||
|
|
||||||
@ -77,7 +77,7 @@ class Container(object):
|
|||||||
|
|
||||||
# Map of relative paths with '/' separators from root of unzipped ePub
|
# Map of relative paths with '/' separators from root of unzipped ePub
|
||||||
# to absolute paths on filesystem with os-specific separators
|
# to absolute paths on filesystem with os-specific separators
|
||||||
opfpath = os.path.abspath(opfpath)
|
opfpath = os.path.abspath(os.path.realpath(opfpath))
|
||||||
for dirpath, _dirnames, filenames in os.walk(self.root):
|
for dirpath, _dirnames, filenames in os.walk(self.root):
|
||||||
for f in filenames:
|
for f in filenames:
|
||||||
path = join(dirpath, f)
|
path = join(dirpath, f)
|
||||||
@ -406,8 +406,9 @@ class Container(object):
|
|||||||
child.get('content', '').strip() in {'{}', ''}):
|
child.get('content', '').strip() in {'{}', ''}):
|
||||||
remove.add(child)
|
remove.add(child)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
continue # Happens for XML comments
|
continue # Happens for XML comments
|
||||||
for child in remove: mdata.remove(child)
|
for child in remove:
|
||||||
|
mdata.remove(child)
|
||||||
if len(mdata) > 0:
|
if len(mdata) > 0:
|
||||||
mdata[-1].tail = '\n '
|
mdata[-1].tail = '\n '
|
||||||
|
|
||||||
@ -473,17 +474,17 @@ class EpubContainer(Container):
|
|||||||
book_type = 'epub'
|
book_type = 'epub'
|
||||||
|
|
||||||
META_INF = {
|
META_INF = {
|
||||||
'container.xml' : True,
|
'container.xml': True,
|
||||||
'manifest.xml' : False,
|
'manifest.xml': False,
|
||||||
'encryption.xml' : False,
|
'encryption.xml': False,
|
||||||
'metadata.xml' : False,
|
'metadata.xml': False,
|
||||||
'signatures.xml' : False,
|
'signatures.xml': False,
|
||||||
'rights.xml' : False,
|
'rights.xml': False,
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, pathtoepub, log):
|
def __init__(self, pathtoepub, log):
|
||||||
self.pathtoepub = pathtoepub
|
self.pathtoepub = pathtoepub
|
||||||
tdir = self.root = PersistentTemporaryDirectory('_epub_container')
|
tdir = self.root = os.path.abspath(os.path.realpath(PersistentTemporaryDirectory('_epub_container')))
|
||||||
with open(self.pathtoepub, 'rb') as stream:
|
with open(self.pathtoepub, 'rb') as stream:
|
||||||
try:
|
try:
|
||||||
zf = ZipFile(stream)
|
zf = ZipFile(stream)
|
||||||
@ -616,7 +617,7 @@ class AZW3Container(Container):
|
|||||||
|
|
||||||
def __init__(self, pathtoazw3, log):
|
def __init__(self, pathtoazw3, log):
|
||||||
self.pathtoazw3 = pathtoazw3
|
self.pathtoazw3 = pathtoazw3
|
||||||
tdir = self.root = PersistentTemporaryDirectory('_azw3_container')
|
tdir = self.root = os.path.abspath(os.path.realpath(PersistentTemporaryDirectory('_azw3_container')))
|
||||||
with open(pathtoazw3, 'rb') as stream:
|
with open(pathtoazw3, 'rb') as stream:
|
||||||
raw = stream.read(3)
|
raw = stream.read(3)
|
||||||
if raw == b'TPZ':
|
if raw == b'TPZ':
|
||||||
@ -670,7 +671,8 @@ class AZW3Container(Container):
|
|||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
def get_container(path, log=None):
|
def get_container(path, log=None):
|
||||||
if log is None: log = default_log
|
if log is None:
|
||||||
|
log = default_log
|
||||||
ebook = (AZW3Container if path.rpartition('.')[-1].lower() in {'azw3', 'mobi'}
|
ebook = (AZW3Container if path.rpartition('.')[-1].lower() in {'azw3', 'mobi'}
|
||||||
else EpubContainer)(path, log)
|
else EpubContainer)(path, log)
|
||||||
return ebook
|
return ebook
|
||||||
|
@ -175,7 +175,7 @@ def gui_polish(data):
|
|||||||
if not data.pop('metadata'):
|
if not data.pop('metadata'):
|
||||||
data.pop('opf')
|
data.pop('opf')
|
||||||
if not data.pop('do_cover'):
|
if not data.pop('do_cover'):
|
||||||
data.pop('cover')
|
data.pop('cover', None)
|
||||||
file_map = {x:x for x in files}
|
file_map = {x:x for x in files}
|
||||||
opts = ALL_OPTS.copy()
|
opts = ALL_OPTS.copy()
|
||||||
opts.update(data)
|
opts.update(data)
|
||||||
|
@ -716,10 +716,11 @@ def choose_save_file(window, name, title, filters=[], all_files=True):
|
|||||||
ans = ans[0]
|
ans = ans[0]
|
||||||
return ans
|
return ans
|
||||||
|
|
||||||
def choose_images(window, name, title, select_only_single_file=True):
|
def choose_images(window, name, title, select_only_single_file=True,
|
||||||
|
formats=('png', 'gif', 'jpg', 'jpeg', 'svg')):
|
||||||
mode = QFileDialog.ExistingFile if select_only_single_file else QFileDialog.ExistingFiles
|
mode = QFileDialog.ExistingFile if select_only_single_file else QFileDialog.ExistingFiles
|
||||||
fd = FileDialog(title=title, name=name,
|
fd = FileDialog(title=title, name=name,
|
||||||
filters=[('Images', ['png', 'gif', 'jpeg', 'jpg', 'svg'])],
|
filters=[('Images', list(formats))],
|
||||||
parent=window, add_all_files_filter=False, mode=mode,
|
parent=window, add_all_files_filter=False, mode=mode,
|
||||||
)
|
)
|
||||||
fd.setParent(None)
|
fd.setParent(None)
|
||||||
|
@ -10,8 +10,7 @@ from functools import partial
|
|||||||
from threading import Thread
|
from threading import Thread
|
||||||
from contextlib import closing
|
from contextlib import closing
|
||||||
|
|
||||||
from PyQt4.Qt import (QToolButton, QDialog, QGridLayout, QIcon, QLabel,
|
from PyQt4.Qt import (QToolButton, QDialog, QGridLayout, QIcon, QLabel, QDialogButtonBox)
|
||||||
QCheckBox, QDialogButtonBox)
|
|
||||||
|
|
||||||
from calibre.gui2.actions import InterfaceAction
|
from calibre.gui2.actions import InterfaceAction
|
||||||
from calibre.gui2 import (error_dialog, Dispatcher, warning_dialog, gprefs,
|
from calibre.gui2 import (error_dialog, Dispatcher, warning_dialog, gprefs,
|
||||||
@ -21,7 +20,7 @@ from calibre.gui2.widgets import HistoryLineEdit
|
|||||||
from calibre.utils.config import prefs, tweaks
|
from calibre.utils.config import prefs, tweaks
|
||||||
from calibre.utils.date import now
|
from calibre.utils.date import now
|
||||||
|
|
||||||
class Worker(Thread): # {{{
|
class Worker(Thread): # {{{
|
||||||
|
|
||||||
def __init__(self, ids, db, loc, progress, done, delete_after):
|
def __init__(self, ids, db, loc, progress, done, delete_after):
|
||||||
Thread.__init__(self)
|
Thread.__init__(self)
|
||||||
@ -71,8 +70,10 @@ class Worker(Thread): # {{{
|
|||||||
mi.timestamp = now()
|
mi.timestamp = now()
|
||||||
self.progress(i, mi.title)
|
self.progress(i, mi.title)
|
||||||
fmts = self.db.formats(x, index_is_id=True)
|
fmts = self.db.formats(x, index_is_id=True)
|
||||||
if not fmts: fmts = []
|
if not fmts:
|
||||||
else: fmts = fmts.split(',')
|
fmts = []
|
||||||
|
else:
|
||||||
|
fmts = fmts.split(',')
|
||||||
paths = []
|
paths = []
|
||||||
for fmt in fmts:
|
for fmt in fmts:
|
||||||
p = self.db.format(x, fmt, index_is_id=True,
|
p = self.db.format(x, fmt, index_is_id=True,
|
||||||
@ -82,7 +83,7 @@ class Worker(Thread): # {{{
|
|||||||
automerged = False
|
automerged = False
|
||||||
if prefs['add_formats_to_existing']:
|
if prefs['add_formats_to_existing']:
|
||||||
identical_book_list = newdb.find_identical_books(mi)
|
identical_book_list = newdb.find_identical_books(mi)
|
||||||
if identical_book_list: # books with same author and nearly same title exist in newdb
|
if identical_book_list: # books with same author and nearly same title exist in newdb
|
||||||
self.auto_merged_ids[x] = _('%(title)s by %(author)s')%\
|
self.auto_merged_ids[x] = _('%(title)s by %(author)s')%\
|
||||||
dict(title=mi.title, author=mi.format_field('authors')[1])
|
dict(title=mi.title, author=mi.format_field('authors')[1])
|
||||||
automerged = True
|
automerged = True
|
||||||
@ -127,7 +128,7 @@ class Worker(Thread): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class ChooseLibrary(QDialog): # {{{
|
class ChooseLibrary(QDialog): # {{{
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
super(ChooseLibrary, self).__init__(parent)
|
super(ChooseLibrary, self).__init__(parent)
|
||||||
@ -146,12 +147,19 @@ class ChooseLibrary(QDialog): # {{{
|
|||||||
b.setToolTip(_('Browse for library'))
|
b.setToolTip(_('Browse for library'))
|
||||||
b.clicked.connect(self.browse)
|
b.clicked.connect(self.browse)
|
||||||
l.addWidget(b, 0, 2)
|
l.addWidget(b, 0, 2)
|
||||||
self.c = c = QCheckBox(_('&Delete after copy'))
|
self.bb = bb = QDialogButtonBox(QDialogButtonBox.Cancel)
|
||||||
l.addWidget(c, 1, 0, 1, 3)
|
|
||||||
self.bb = bb = QDialogButtonBox(QDialogButtonBox.Ok|QDialogButtonBox.Cancel)
|
|
||||||
bb.accepted.connect(self.accept)
|
bb.accepted.connect(self.accept)
|
||||||
bb.rejected.connect(self.reject)
|
bb.rejected.connect(self.reject)
|
||||||
l.addWidget(bb, 2, 0, 1, 3)
|
self.delete_after_copy = False
|
||||||
|
b = bb.addButton(_('&Copy'), bb.AcceptRole)
|
||||||
|
b.setIcon(QIcon(I('edit-copy.png')))
|
||||||
|
b.setToolTip(_('Copy to the specified library'))
|
||||||
|
b2 = bb.addButton(_('&Move'), bb.AcceptRole)
|
||||||
|
b2.clicked.connect(lambda: setattr(self, 'delete_after_copy', True))
|
||||||
|
b2.setIcon(QIcon(I('edit-cut.png')))
|
||||||
|
b2.setToolTip(_('Copy to the specified library and delete from the current library'))
|
||||||
|
b.setDefault(True)
|
||||||
|
l.addWidget(bb, 1, 0, 1, 3)
|
||||||
le.setMinimumWidth(350)
|
le.setMinimumWidth(350)
|
||||||
self.resize(self.sizeHint())
|
self.resize(self.sizeHint())
|
||||||
|
|
||||||
@ -163,7 +171,7 @@ class ChooseLibrary(QDialog): # {{{
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def args(self):
|
def args(self):
|
||||||
return (unicode(self.le.text()), self.c.isChecked())
|
return (unicode(self.le.text()), self.delete_after_copy)
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class CopyToLibraryAction(InterfaceAction):
|
class CopyToLibraryAction(InterfaceAction):
|
||||||
@ -204,7 +212,7 @@ class CopyToLibraryAction(InterfaceAction):
|
|||||||
self.menu.addAction(name, partial(self.copy_to_library,
|
self.menu.addAction(name, partial(self.copy_to_library,
|
||||||
loc))
|
loc))
|
||||||
self.menu.addAction(name + ' ' + _('(delete after copy)'),
|
self.menu.addAction(name + ' ' + _('(delete after copy)'),
|
||||||
partial(self.copy_to_library, loc, delete_after=True))
|
partial(self.copy_to_library, loc, delete_after=True))
|
||||||
self.menu.addSeparator()
|
self.menu.addSeparator()
|
||||||
|
|
||||||
self.menu.addAction(_('Choose library by path...'), self.choose_library)
|
self.menu.addAction(_('Choose library by path...'), self.choose_library)
|
||||||
@ -214,6 +222,8 @@ class CopyToLibraryAction(InterfaceAction):
|
|||||||
d = ChooseLibrary(self.gui)
|
d = ChooseLibrary(self.gui)
|
||||||
if d.exec_() == d.Accepted:
|
if d.exec_() == d.Accepted:
|
||||||
path, delete_after = d.args
|
path, delete_after = d.args
|
||||||
|
if not path:
|
||||||
|
return
|
||||||
db = self.gui.library_view.model().db
|
db = self.gui.library_view.model().db
|
||||||
current = os.path.normcase(os.path.abspath(db.library_path))
|
current = os.path.normcase(os.path.abspath(db.library_path))
|
||||||
if current == os.path.normcase(os.path.abspath(path)):
|
if current == os.path.normcase(os.path.abspath(path)):
|
||||||
|
@ -17,14 +17,14 @@ from PyQt4.Qt import (QDialog, QGridLayout, QIcon, QCheckBox, QLabel, QFrame,
|
|||||||
QSizePolicy, QTimer, QModelIndex, QTextEdit,
|
QSizePolicy, QTimer, QModelIndex, QTextEdit,
|
||||||
QInputDialog, QMenu)
|
QInputDialog, QMenu)
|
||||||
|
|
||||||
from calibre.gui2 import error_dialog, Dispatcher, gprefs
|
from calibre.gui2 import error_dialog, Dispatcher, gprefs, question_dialog
|
||||||
from calibre.gui2.actions import InterfaceAction
|
from calibre.gui2.actions import InterfaceAction
|
||||||
from calibre.gui2.convert.metadata import create_opf_file
|
from calibre.gui2.convert.metadata import create_opf_file
|
||||||
from calibre.gui2.dialogs.progress import ProgressDialog
|
from calibre.gui2.dialogs.progress import ProgressDialog
|
||||||
from calibre.ptempfile import PersistentTemporaryDirectory
|
from calibre.ptempfile import PersistentTemporaryDirectory
|
||||||
from calibre.utils.config_base import tweaks
|
from calibre.utils.config_base import tweaks
|
||||||
|
|
||||||
class Polish(QDialog): # {{{
|
class Polish(QDialog): # {{{
|
||||||
|
|
||||||
def __init__(self, db, book_id_map, parent=None):
|
def __init__(self, db, book_id_map, parent=None):
|
||||||
from calibre.ebooks.oeb.polish.main import HELP
|
from calibre.ebooks.oeb.polish.main import HELP
|
||||||
@ -58,7 +58,7 @@ class Polish(QDialog): # {{{
|
|||||||
' formats are not capable of supporting all the'
|
' formats are not capable of supporting all the'
|
||||||
' metadata in calibre.</p><p>There is a separate option to'
|
' metadata in calibre.</p><p>There is a separate option to'
|
||||||
' update the cover.</p>'),
|
' update the cover.</p>'),
|
||||||
'do_cover': _('<p>Update the covers in the ebook files to match the'
|
'do_cover': _('<p>Update the covers in the ebook files to match the'
|
||||||
' current cover in the calibre library.</p>'
|
' current cover in the calibre library.</p>'
|
||||||
'<p>If the ebook file does not have'
|
'<p>If the ebook file does not have'
|
||||||
' an identifiable cover, a new cover is inserted.</p>'
|
' an identifiable cover, a new cover is inserted.</p>'
|
||||||
@ -204,6 +204,15 @@ class Polish(QDialog): # {{{
|
|||||||
ac[action] = saved_prefs[action] = bool(getattr(self, 'opt_'+action).isChecked())
|
ac[action] = saved_prefs[action] = bool(getattr(self, 'opt_'+action).isChecked())
|
||||||
if ac[action]:
|
if ac[action]:
|
||||||
something = True
|
something = True
|
||||||
|
if ac['jacket'] and not ac['metadata']:
|
||||||
|
if not question_dialog(self, _('Must update metadata'),
|
||||||
|
_('You have selected the option to add metadata as '
|
||||||
|
'a "book jacket". For this option to work, you '
|
||||||
|
'must also select the option to update metadata in'
|
||||||
|
' the book files. Do you want to select it?')):
|
||||||
|
return
|
||||||
|
ac['metadata'] = saved_prefs['metadata'] = True
|
||||||
|
self.opt_metadata.setChecked(True)
|
||||||
if not something:
|
if not something:
|
||||||
return error_dialog(self, _('No actions selected'),
|
return error_dialog(self, _('No actions selected'),
|
||||||
_('You must select at least one action, or click Cancel.'),
|
_('You must select at least one action, or click Cancel.'),
|
||||||
@ -275,7 +284,7 @@ class Polish(QDialog): # {{{
|
|||||||
self.jobs.append((desc, data, book_id, base, is_orig))
|
self.jobs.append((desc, data, book_id, base, is_orig))
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class Report(QDialog): # {{{
|
class Report(QDialog): # {{{
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
QDialog.__init__(self, parent)
|
QDialog.__init__(self, parent)
|
||||||
@ -427,7 +436,7 @@ class PolishAction(InterfaceAction):
|
|||||||
supported = set(SUPPORTED)
|
supported = set(SUPPORTED)
|
||||||
for x in SUPPORTED:
|
for x in SUPPORTED:
|
||||||
supported.add('ORIGINAL_'+x)
|
supported.add('ORIGINAL_'+x)
|
||||||
ans = [(x, set( (db.formats(x, index_is_id=True) or '').split(',') )
|
ans = [(x, set((db.formats(x, index_is_id=True) or '').split(','))
|
||||||
.intersection(supported)) for x in book_ids]
|
.intersection(supported)) for x in book_ids]
|
||||||
ans = [x for x in ans if x[1]]
|
ans = [x for x in ans if x[1]]
|
||||||
if not ans:
|
if not ans:
|
||||||
@ -476,8 +485,7 @@ class PolishAction(InterfaceAction):
|
|||||||
db.save_original_format(book_id, fmt, notify=False)
|
db.save_original_format(book_id, fmt, notify=False)
|
||||||
with open(path, 'rb') as f:
|
with open(path, 'rb') as f:
|
||||||
db.add_format(book_id, fmt, f, index_is_id=True)
|
db.add_format(book_id, fmt, f, index_is_id=True)
|
||||||
self.gui.status_bar.show_message(job.description + \
|
self.gui.status_bar.show_message(job.description + (' completed'), 2000)
|
||||||
(' completed'), 2000)
|
|
||||||
try:
|
try:
|
||||||
shutil.rmtree(base)
|
shutil.rmtree(base)
|
||||||
parent = os.path.dirname(base)
|
parent = os.path.dirname(base)
|
||||||
|
@ -16,11 +16,10 @@ from calibre.constants import __appname__
|
|||||||
from calibre.gui2.search_box import SearchBox2, SavedSearchBox
|
from calibre.gui2.search_box import SearchBox2, SavedSearchBox
|
||||||
from calibre.gui2.throbber import ThrobbingButton
|
from calibre.gui2.throbber import ThrobbingButton
|
||||||
from calibre.gui2.bars import BarsManager
|
from calibre.gui2.bars import BarsManager
|
||||||
from calibre.gui2.widgets import ComboBoxWithHelp
|
|
||||||
from calibre.utils.config_base import tweaks
|
from calibre.utils.config_base import tweaks
|
||||||
from calibre import human_readable
|
from calibre import human_readable
|
||||||
|
|
||||||
class LocationManager(QObject): # {{{
|
class LocationManager(QObject): # {{{
|
||||||
|
|
||||||
locations_changed = pyqtSignal()
|
locations_changed = pyqtSignal()
|
||||||
unmount_device = pyqtSignal()
|
unmount_device = pyqtSignal()
|
||||||
@ -165,7 +164,7 @@ class LocationManager(QObject): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class SearchBar(QWidget): # {{{
|
class SearchBar(QWidget): # {{{
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
QWidget.__init__(self, parent)
|
QWidget.__init__(self, parent)
|
||||||
@ -173,11 +172,13 @@ class SearchBar(QWidget): # {{{
|
|||||||
self.setLayout(self._layout)
|
self.setLayout(self._layout)
|
||||||
self._layout.setContentsMargins(0,5,0,0)
|
self._layout.setContentsMargins(0,5,0,0)
|
||||||
|
|
||||||
x = ComboBoxWithHelp(self)
|
x = QToolButton(self)
|
||||||
x.setMaximumSize(QSize(150, 16777215))
|
x.setText(_('Virtual Library'))
|
||||||
x.setObjectName("search_restriction")
|
x.setIcon(QIcon(I('lt.png')))
|
||||||
|
x.setObjectName("virtual_library")
|
||||||
|
x.setToolButtonStyle(Qt.ToolButtonTextBesideIcon)
|
||||||
l.addWidget(x)
|
l.addWidget(x)
|
||||||
parent.search_restriction = x
|
parent.virtual_library = x
|
||||||
|
|
||||||
x = QLabel(self)
|
x = QLabel(self)
|
||||||
x.setObjectName("search_count")
|
x.setObjectName("search_count")
|
||||||
@ -243,7 +244,7 @@ class SearchBar(QWidget): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class Spacer(QWidget): # {{{
|
class Spacer(QWidget): # {{{
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
QWidget.__init__(self, parent)
|
QWidget.__init__(self, parent)
|
||||||
@ -252,7 +253,7 @@ class Spacer(QWidget): # {{{
|
|||||||
self.l.addStretch(10)
|
self.l.addStretch(10)
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class MainWindowMixin(object): # {{{
|
class MainWindowMixin(object): # {{{
|
||||||
|
|
||||||
def __init__(self, db):
|
def __init__(self, db):
|
||||||
self.setObjectName('MainWindow')
|
self.setObjectName('MainWindow')
|
||||||
|
@ -927,9 +927,9 @@ class Cover(ImageView): # {{{
|
|||||||
return sz
|
return sz
|
||||||
|
|
||||||
def select_cover(self, *args):
|
def select_cover(self, *args):
|
||||||
files = choose_images(self, 'change cover dialog',
|
files = choose_images(
|
||||||
_('Choose cover for ') +
|
self, 'change cover dialog', _('Choose cover for ') + self.dialog.title.current_val,
|
||||||
self.dialog.title.current_val)
|
formats=('png', 'gif', 'jpg', 'jpeg'))
|
||||||
if not files:
|
if not files:
|
||||||
return
|
return
|
||||||
_file = files[0]
|
_file = files[0]
|
||||||
|
@ -14,7 +14,6 @@ from calibre.gui2.preferences.behavior_ui import Ui_Form
|
|||||||
from calibre.gui2 import config, info_dialog, dynamic, gprefs
|
from calibre.gui2 import config, info_dialog, dynamic, gprefs
|
||||||
from calibre.utils.config import prefs
|
from calibre.utils.config import prefs
|
||||||
from calibre.customize.ui import available_output_formats, all_input_formats
|
from calibre.customize.ui import available_output_formats, all_input_formats
|
||||||
from calibre.utils.search_query_parser import saved_searches
|
|
||||||
from calibre.ebooks import BOOK_EXTENSIONS
|
from calibre.ebooks import BOOK_EXTENSIONS
|
||||||
from calibre.ebooks.oeb.iterator import is_supported
|
from calibre.ebooks.oeb.iterator import is_supported
|
||||||
from calibre.constants import iswindows
|
from calibre.constants import iswindows
|
||||||
@ -48,9 +47,13 @@ class ConfigWidget(ConfigWidgetBase, Ui_Form):
|
|||||||
choices = [(x.upper(), x) for x in output_formats]
|
choices = [(x.upper(), x) for x in output_formats]
|
||||||
r('output_format', prefs, choices=choices, setting=OutputFormatSetting)
|
r('output_format', prefs, choices=choices, setting=OutputFormatSetting)
|
||||||
|
|
||||||
restrictions = sorted(saved_searches().names(), key=sort_key)
|
restrictions = sorted(db.prefs['virtual_libraries'].iterkeys(), key=sort_key)
|
||||||
choices = [('', '')] + [(x, x) for x in restrictions]
|
choices = [('', '')] + [(x, x) for x in restrictions]
|
||||||
r('gui_restriction', db.prefs, choices=choices)
|
# check that the virtual library still exists
|
||||||
|
vls = db.prefs['virtual_lib_on_startup']
|
||||||
|
if vls and vls not in restrictions:
|
||||||
|
db.prefs['virtual_lib_on_startup'] = ''
|
||||||
|
r('virtual_lib_on_startup', db.prefs, choices=choices)
|
||||||
self.reset_confirmation_button.clicked.connect(self.reset_confirmation_dialogs)
|
self.reset_confirmation_button.clicked.connect(self.reset_confirmation_dialogs)
|
||||||
|
|
||||||
self.input_up_button.clicked.connect(self.up_input)
|
self.input_up_button.clicked.connect(self.up_input)
|
||||||
|
@ -147,15 +147,15 @@ If not checked, the values can be Yes or No.</string>
|
|||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="label_170">
|
<widget class="QLabel" name="label_170">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Restriction to apply when the current library is opened:</string>
|
<string>Virtual library to apply when the current library is opened:</string>
|
||||||
</property>
|
</property>
|
||||||
<property name="buddy">
|
<property name="buddy">
|
||||||
<cstring>opt_gui_restriction</cstring>
|
<cstring>opt_virtual_lib_on_startup</cstring>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QComboBox" name="opt_gui_restriction">
|
<widget class="QComboBox" name="opt_virtual_lib_on_startup">
|
||||||
<property name="maximumSize">
|
<property name="maximumSize">
|
||||||
<size>
|
<size>
|
||||||
<width>250</width>
|
<width>250</width>
|
||||||
@ -163,7 +163,7 @@ If not checked, the values can be Yes or No.</string>
|
|||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
<property name="toolTip">
|
<property name="toolTip">
|
||||||
<string>Apply this restriction on calibre startup if the current library is being used. Also applied when switching to this library. Note that this setting is per library. </string>
|
<string>Use this virtual library on calibre startup if the current library is being used. Also applied when switching to this library. Note that this setting is per library. </string>
|
||||||
</property>
|
</property>
|
||||||
<property name="sizeAdjustPolicy">
|
<property name="sizeAdjustPolicy">
|
||||||
<enum>QComboBox::AdjustToMinimumContentsLengthWithIcon</enum>
|
<enum>QComboBox::AdjustToMinimumContentsLengthWithIcon</enum>
|
||||||
|
@ -12,7 +12,6 @@ from PyQt4.Qt import Qt, QUrl, QDialog, QSize, QVBoxLayout, QLabel, \
|
|||||||
|
|
||||||
from calibre.gui2.preferences import ConfigWidgetBase, test_widget
|
from calibre.gui2.preferences import ConfigWidgetBase, test_widget
|
||||||
from calibre.gui2.preferences.server_ui import Ui_Form
|
from calibre.gui2.preferences.server_ui import Ui_Form
|
||||||
from calibre.utils.search_query_parser import saved_searches
|
|
||||||
from calibre.library.server import server_config
|
from calibre.library.server import server_config
|
||||||
from calibre.utils.config import ConfigProxy
|
from calibre.utils.config import ConfigProxy
|
||||||
from calibre.gui2 import error_dialog, config, open_url, warning_dialog, \
|
from calibre.gui2 import error_dialog, config, open_url, warning_dialog, \
|
||||||
@ -44,13 +43,13 @@ class ConfigWidget(ConfigWidgetBase, Ui_Form):
|
|||||||
else self.opt_password.Password))
|
else self.opt_password.Password))
|
||||||
self.opt_password.setEchoMode(self.opt_password.Password)
|
self.opt_password.setEchoMode(self.opt_password.Password)
|
||||||
|
|
||||||
restrictions = sorted(saved_searches().names(), key=sort_key)
|
restrictions = sorted(db.prefs['virtual_libraries'].iterkeys(), key=sort_key)
|
||||||
# verify that the current restriction still exists. If not, clear it.
|
|
||||||
csr = db.prefs.get('cs_restriction', None)
|
|
||||||
if csr and csr not in restrictions:
|
|
||||||
db.prefs.set('cs_restriction', '')
|
|
||||||
choices = [('', '')] + [(x, x) for x in restrictions]
|
choices = [('', '')] + [(x, x) for x in restrictions]
|
||||||
r('cs_restriction', db.prefs, choices=choices)
|
# check that the virtual library still exists
|
||||||
|
vls = db.prefs['cs_virtual_lib_on_startup']
|
||||||
|
if vls and vls not in restrictions:
|
||||||
|
db.prefs['cs_virtual_lib_on_startup'] = ''
|
||||||
|
r('cs_virtual_lib_on_startup', db.prefs, choices=choices)
|
||||||
|
|
||||||
self.start_button.setEnabled(not getattr(self.server, 'is_running', False))
|
self.start_button.setEnabled(not getattr(self.server, 'is_running', False))
|
||||||
self.test_button.setEnabled(not self.start_button.isEnabled())
|
self.test_button.setEnabled(not self.start_button.isEnabled())
|
||||||
|
@ -139,14 +139,14 @@
|
|||||||
<item row="7" column="0">
|
<item row="7" column="0">
|
||||||
<widget class="QLabel" name="label_164">
|
<widget class="QLabel" name="label_164">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Restriction (saved search) to apply:</string>
|
<string>Virtual library to apply:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="7" column="1" colspan="2">
|
<item row="7" column="1" colspan="2">
|
||||||
<widget class="QComboBox" name="opt_cs_restriction">
|
<widget class="QComboBox" name="opt_cs_virtual_lib_on_startup">
|
||||||
<property name="toolTip">
|
<property name="toolTip">
|
||||||
<string>This restriction (based on a saved search) will restrict the books the content server makes available to those matching the search. This setting is per library (i.e. you can have a different restriction per library).</string>
|
<string>Setting a virtual library will restrict the books the content server makes available to those in the library. This setting is per library (i.e. you can have a different value per library).</string>
|
||||||
</property>
|
</property>
|
||||||
<property name="sizeAdjustPolicy">
|
<property name="sizeAdjustPolicy">
|
||||||
<enum>QComboBox::AdjustToMinimumContentsLengthWithIcon</enum>
|
<enum>QComboBox::AdjustToMinimumContentsLengthWithIcon</enum>
|
||||||
|
@ -19,9 +19,8 @@ from calibre.gui2.dialogs.confirm_delete import confirm
|
|||||||
from calibre.gui2.dialogs.saved_search_editor import SavedSearchEditor
|
from calibre.gui2.dialogs.saved_search_editor import SavedSearchEditor
|
||||||
from calibre.gui2.dialogs.search import SearchDialog
|
from calibre.gui2.dialogs.search import SearchDialog
|
||||||
from calibre.utils.search_query_parser import saved_searches
|
from calibre.utils.search_query_parser import saved_searches
|
||||||
from calibre.utils.icu import sort_key
|
|
||||||
|
|
||||||
class SearchLineEdit(QLineEdit): # {{{
|
class SearchLineEdit(QLineEdit): # {{{
|
||||||
key_pressed = pyqtSignal(object)
|
key_pressed = pyqtSignal(object)
|
||||||
|
|
||||||
def keyPressEvent(self, event):
|
def keyPressEvent(self, event):
|
||||||
@ -42,7 +41,7 @@ class SearchLineEdit(QLineEdit): # {{{
|
|||||||
return QLineEdit.paste(self)
|
return QLineEdit.paste(self)
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class SearchBox2(QComboBox): # {{{
|
class SearchBox2(QComboBox): # {{{
|
||||||
|
|
||||||
'''
|
'''
|
||||||
To use this class:
|
To use this class:
|
||||||
@ -59,7 +58,7 @@ class SearchBox2(QComboBox): # {{{
|
|||||||
accurate.
|
accurate.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
INTERVAL = 1500 #: Time to wait before emitting search signal
|
INTERVAL = 1500 #: Time to wait before emitting search signal
|
||||||
MAX_COUNT = 25
|
MAX_COUNT = 25
|
||||||
|
|
||||||
search = pyqtSignal(object)
|
search = pyqtSignal(object)
|
||||||
@ -254,7 +253,7 @@ class SearchBox2(QComboBox): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class SavedSearchBox(QComboBox): # {{{
|
class SavedSearchBox(QComboBox): # {{{
|
||||||
|
|
||||||
'''
|
'''
|
||||||
To use this class:
|
To use this class:
|
||||||
@ -332,6 +331,10 @@ class SavedSearchBox(QComboBox): # {{{
|
|||||||
name = unicode(self.currentText())
|
name = unicode(self.currentText())
|
||||||
if not name.strip():
|
if not name.strip():
|
||||||
name = unicode(self.search_box.text()).replace('"', '')
|
name = unicode(self.search_box.text()).replace('"', '')
|
||||||
|
if not (name and self.search_box.text()):
|
||||||
|
error_dialog(self, _('Create saved search'),
|
||||||
|
_('There is no search to save'), show=True)
|
||||||
|
return
|
||||||
saved_searches().delete(name)
|
saved_searches().delete(name)
|
||||||
saved_searches().add(name, unicode(self.search_box.text()))
|
saved_searches().add(name, unicode(self.search_box.text()))
|
||||||
# now go through an initialization cycle to ensure that the combobox has
|
# now go through an initialization cycle to ensure that the combobox has
|
||||||
@ -339,7 +342,7 @@ class SavedSearchBox(QComboBox): # {{{
|
|||||||
# references the new search instead of the text in the search.
|
# references the new search instead of the text in the search.
|
||||||
self.clear()
|
self.clear()
|
||||||
self.setCurrentIndex(self.findText(name))
|
self.setCurrentIndex(self.findText(name))
|
||||||
self.saved_search_selected (name)
|
self.saved_search_selected(name)
|
||||||
self.changed.emit()
|
self.changed.emit()
|
||||||
|
|
||||||
def delete_current_search(self):
|
def delete_current_search(self):
|
||||||
@ -361,15 +364,15 @@ class SavedSearchBox(QComboBox): # {{{
|
|||||||
self.changed.emit()
|
self.changed.emit()
|
||||||
|
|
||||||
# SIGNALed from the main UI
|
# SIGNALed from the main UI
|
||||||
def copy_search_button_clicked (self):
|
def copy_search_button_clicked(self):
|
||||||
idx = self.currentIndex();
|
idx = self.currentIndex()
|
||||||
if idx < 0:
|
if idx < 0:
|
||||||
return
|
return
|
||||||
self.search_box.set_search_string(saved_searches().lookup(unicode(self.currentText())))
|
self.search_box.set_search_string(saved_searches().lookup(unicode(self.currentText())))
|
||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class SearchBoxMixin(object): # {{{
|
class SearchBoxMixin(object): # {{{
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.search.initialize('main_search_history', colorize=True,
|
self.search.initialize('main_search_history', colorize=True,
|
||||||
@ -443,7 +446,7 @@ class SearchBoxMixin(object): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class SavedSearchBoxMixin(object): # {{{
|
class SavedSearchBoxMixin(object): # {{{
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.saved_search.changed.connect(self.saved_searches_changed)
|
self.saved_search.changed.connect(self.saved_searches_changed)
|
||||||
@ -452,7 +455,7 @@ class SavedSearchBoxMixin(object): # {{{
|
|||||||
self.saved_search.save_search_button_clicked)
|
self.saved_search.save_search_button_clicked)
|
||||||
self.copy_search_button.clicked.connect(
|
self.copy_search_button.clicked.connect(
|
||||||
self.saved_search.copy_search_button_clicked)
|
self.saved_search.copy_search_button_clicked)
|
||||||
self.saved_searches_changed()
|
# self.saved_searches_changed()
|
||||||
self.saved_search.initialize(self.search, colorize=True,
|
self.saved_search.initialize(self.search, colorize=True,
|
||||||
help_text=_('Saved Searches'))
|
help_text=_('Saved Searches'))
|
||||||
self.saved_search.setToolTip(
|
self.saved_search.setToolTip(
|
||||||
@ -479,18 +482,10 @@ class SavedSearchBoxMixin(object): # {{{
|
|||||||
partial(self.do_saved_search_edit, None))
|
partial(self.do_saved_search_edit, None))
|
||||||
|
|
||||||
def saved_searches_changed(self, set_restriction=None, recount=True):
|
def saved_searches_changed(self, set_restriction=None, recount=True):
|
||||||
p = sorted(saved_searches().names(), key=sort_key)
|
self.build_search_restriction_list()
|
||||||
if set_restriction is None:
|
|
||||||
set_restriction = unicode(self.search_restriction.currentText())
|
|
||||||
# rebuild the restrictions combobox using current saved searches
|
|
||||||
self.search_restriction.clear()
|
|
||||||
self.search_restriction.addItem('')
|
|
||||||
self.search_restriction.addItem(_('*Current search'))
|
|
||||||
if recount:
|
if recount:
|
||||||
self.tags_view.recount()
|
self.tags_view.recount()
|
||||||
for s in p:
|
if set_restriction: # redo the search restriction if there was one
|
||||||
self.search_restriction.addItem(s)
|
|
||||||
if set_restriction: # redo the search restriction if there was one
|
|
||||||
self.apply_named_search_restriction(set_restriction)
|
self.apply_named_search_restriction(set_restriction)
|
||||||
|
|
||||||
def do_saved_search_edit(self, search):
|
def do_saved_search_edit(self, search):
|
||||||
|
@ -4,23 +4,505 @@ Created on 10 Jun 2010
|
|||||||
@author: charles
|
@author: charles
|
||||||
'''
|
'''
|
||||||
|
|
||||||
from PyQt4.Qt import Qt
|
from functools import partial
|
||||||
|
|
||||||
|
from PyQt4.Qt import (
|
||||||
|
Qt, QMenu, QPoint, QIcon, QDialog, QGridLayout, QLabel, QLineEdit, QComboBox,
|
||||||
|
QDialogButtonBox, QSize, QVBoxLayout, QListWidget, QStringList, QCheckBox)
|
||||||
|
|
||||||
|
from calibre.gui2 import error_dialog, question_dialog
|
||||||
|
from calibre.gui2.widgets import ComboBoxWithHelp
|
||||||
|
from calibre.utils.icu import sort_key
|
||||||
|
from calibre.utils.pyparsing import ParseException
|
||||||
|
from calibre.utils.search_query_parser import saved_searches
|
||||||
|
|
||||||
|
class SelectNames(QDialog): # {{{
|
||||||
|
|
||||||
|
def __init__(self, names, txt, parent=None):
|
||||||
|
QDialog.__init__(self, parent)
|
||||||
|
self.l = l = QVBoxLayout(self)
|
||||||
|
self.setLayout(l)
|
||||||
|
|
||||||
|
self.la = la = QLabel(_('Create a Virtual Library based on %s') % txt)
|
||||||
|
l.addWidget(la)
|
||||||
|
|
||||||
|
self._names = QListWidget(self)
|
||||||
|
self._names.addItems(QStringList(sorted(names, key=sort_key)))
|
||||||
|
self._names.setSelectionMode(self._names.ExtendedSelection)
|
||||||
|
l.addWidget(self._names)
|
||||||
|
|
||||||
|
self._and = QCheckBox(_('Match all selected %s names')%txt)
|
||||||
|
l.addWidget(self._and)
|
||||||
|
|
||||||
|
self.bb = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
|
||||||
|
self.bb.accepted.connect(self.accept)
|
||||||
|
self.bb.rejected.connect(self.reject)
|
||||||
|
l.addWidget(self.bb)
|
||||||
|
|
||||||
|
self.resize(self.sizeHint())
|
||||||
|
|
||||||
|
@property
|
||||||
|
def names(self):
|
||||||
|
for item in self._names.selectedItems():
|
||||||
|
yield unicode(item.data(Qt.DisplayRole).toString())
|
||||||
|
|
||||||
|
@property
|
||||||
|
def match_type(self):
|
||||||
|
return ' and ' if self._and.isChecked() else ' or '
|
||||||
|
|
||||||
|
# }}}
|
||||||
|
|
||||||
|
MAX_VIRTUAL_LIBRARY_NAME_LENGTH = 40
|
||||||
|
|
||||||
|
def _build_full_search_string(gui):
|
||||||
|
search_templates = (
|
||||||
|
'',
|
||||||
|
'{cl}',
|
||||||
|
'{cr}',
|
||||||
|
'(({cl}) and ({cr}))',
|
||||||
|
'{sb}',
|
||||||
|
'(({cl}) and ({sb}))',
|
||||||
|
'(({cr}) and ({sb}))',
|
||||||
|
'(({cl}) and ({cr}) and ({sb}))'
|
||||||
|
)
|
||||||
|
|
||||||
|
sb = gui.search.current_text
|
||||||
|
db = gui.current_db
|
||||||
|
cr = db.data.get_search_restriction()
|
||||||
|
cl = db.data.get_base_restriction()
|
||||||
|
dex = 0
|
||||||
|
if sb:
|
||||||
|
dex += 4
|
||||||
|
if cr:
|
||||||
|
dex += 2
|
||||||
|
if cl:
|
||||||
|
dex += 1
|
||||||
|
template = search_templates[dex]
|
||||||
|
return template.format(cl=cl, cr=cr, sb=sb).strip()
|
||||||
|
|
||||||
|
class CreateVirtualLibrary(QDialog): # {{{
|
||||||
|
|
||||||
|
def __init__(self, gui, existing_names, editing=None):
|
||||||
|
QDialog.__init__(self, gui)
|
||||||
|
|
||||||
|
self.gui = gui
|
||||||
|
self.existing_names = existing_names
|
||||||
|
|
||||||
|
if editing:
|
||||||
|
self.setWindowTitle(_('Edit virtual library'))
|
||||||
|
else:
|
||||||
|
self.setWindowTitle(_('Create virtual library'))
|
||||||
|
self.setWindowIcon(QIcon(I('lt.png')))
|
||||||
|
|
||||||
|
gl = QGridLayout()
|
||||||
|
self.setLayout(gl)
|
||||||
|
self.la1 = la1 = QLabel(_('Virtual library &name:'))
|
||||||
|
gl.addWidget(la1, 0, 0)
|
||||||
|
self.vl_name = QComboBox()
|
||||||
|
self.vl_name.setEditable(True)
|
||||||
|
self.vl_name.lineEdit().setMaxLength(MAX_VIRTUAL_LIBRARY_NAME_LENGTH)
|
||||||
|
la1.setBuddy(self.vl_name)
|
||||||
|
gl.addWidget(self.vl_name, 0, 1)
|
||||||
|
self.editing = editing
|
||||||
|
|
||||||
|
self.saved_searches_label = QLabel('')
|
||||||
|
self.saved_searches_label.setTextInteractionFlags(Qt.TextSelectableByMouse)
|
||||||
|
gl.addWidget(self.saved_searches_label, 2, 0, 1, 2)
|
||||||
|
|
||||||
|
self.la2 = la2 = QLabel(_('&Search expression:'))
|
||||||
|
gl.addWidget(la2, 1, 0)
|
||||||
|
self.vl_text = QLineEdit()
|
||||||
|
self.vl_text.textChanged.connect(self.search_text_changed)
|
||||||
|
la2.setBuddy(self.vl_text)
|
||||||
|
gl.addWidget(self.vl_text, 1, 1)
|
||||||
|
self.vl_text.setText(_build_full_search_string(self.gui))
|
||||||
|
|
||||||
|
self.sl = sl = QLabel('<p>'+_('Create a virtual library based on: ')+
|
||||||
|
('<a href="author.{0}">{0}</a>, '
|
||||||
|
'<a href="tag.{1}">{1}</a>, '
|
||||||
|
'<a href="publisher.{2}">{2}</a>, '
|
||||||
|
'<a href="series.{3}">{3}</a>, '
|
||||||
|
'<a href="search.{4}">{4}</a>.').format(_('Authors'), _('Tags'),
|
||||||
|
_('Publishers'), _('Series'), _('Saved Searches')))
|
||||||
|
sl.setWordWrap(True)
|
||||||
|
sl.setTextInteractionFlags(Qt.LinksAccessibleByMouse)
|
||||||
|
sl.linkActivated.connect(self.link_activated)
|
||||||
|
gl.addWidget(sl, 3, 0, 1, 2)
|
||||||
|
gl.setRowStretch(3,10)
|
||||||
|
|
||||||
|
self.hl = hl = QLabel(_('''
|
||||||
|
<h2>Virtual Libraries</h2>
|
||||||
|
|
||||||
|
<p>Using <i>virtual libraries</i> you can restrict calibre to only show
|
||||||
|
you books that match a search. When a virtual library is in effect, calibre
|
||||||
|
behaves as though the library contains only the matched books. The Tag Browser
|
||||||
|
display only the tags/authors/series/etc. that belong to the matched books and any searches
|
||||||
|
you do will only search within the books in the virtual library. This
|
||||||
|
is a good way to partition your large library into smaller and easier to work with subsets.</p>
|
||||||
|
|
||||||
|
<p>For example you can use a Virtual Library to only show you books with the Tag <i>"Unread"</i>
|
||||||
|
or only books by <i>"My Favorite Author"</i> or only books in a particular series.</p>
|
||||||
|
'''))
|
||||||
|
hl.setWordWrap(True)
|
||||||
|
hl.setFrameStyle(hl.StyledPanel)
|
||||||
|
gl.addWidget(hl, 0, 3, 4, 1)
|
||||||
|
|
||||||
|
bb = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
|
||||||
|
bb.accepted.connect(self.accept)
|
||||||
|
bb.rejected.connect(self.reject)
|
||||||
|
gl.addWidget(bb, 4, 0, 1, 0)
|
||||||
|
|
||||||
|
if editing:
|
||||||
|
db = self.gui.current_db
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
for dex,vl in enumerate(sorted(virt_libs.keys(), key=sort_key)):
|
||||||
|
self.vl_name.addItem(vl, virt_libs.get(vl, ''))
|
||||||
|
if vl == editing:
|
||||||
|
self.vl_name.setCurrentIndex(dex)
|
||||||
|
self.original_index = dex
|
||||||
|
self.original_search = virt_libs.get(editing, '')
|
||||||
|
self.vl_text.setText(self.original_search)
|
||||||
|
self.new_name = editing
|
||||||
|
self.vl_name.currentIndexChanged[int].connect(self.name_index_changed)
|
||||||
|
self.vl_name.lineEdit().textEdited.connect(self.name_text_edited)
|
||||||
|
|
||||||
|
self.resize(self.sizeHint()+QSize(150, 25))
|
||||||
|
|
||||||
|
def search_text_changed(self, txt):
|
||||||
|
searches = [_('Saved searches recognized in the expression:')]
|
||||||
|
txt = unicode(txt)
|
||||||
|
while txt:
|
||||||
|
p = txt.partition('search:')
|
||||||
|
if p[1]: # found 'search:'
|
||||||
|
possible_search = p[2]
|
||||||
|
if possible_search: # something follows the 'search:'
|
||||||
|
if possible_search[0] == '"': # strip any quotes
|
||||||
|
possible_search = possible_search[1:].partition('"')
|
||||||
|
else: # find end of the search name. Is EOL, space, rparen
|
||||||
|
sp = possible_search.find(' ')
|
||||||
|
pp = possible_search.find(')')
|
||||||
|
if pp < 0 or (sp > 0 and sp <= pp):
|
||||||
|
# space in string before rparen, or neither found
|
||||||
|
possible_search = possible_search.partition(' ')
|
||||||
|
else:
|
||||||
|
# rparen in string before space
|
||||||
|
possible_search = possible_search.partition(')')
|
||||||
|
txt = possible_search[2] # grab remainder of the string
|
||||||
|
search_name = possible_search[0]
|
||||||
|
if search_name.startswith('='):
|
||||||
|
search_name = search_name[1:]
|
||||||
|
if search_name in saved_searches().names():
|
||||||
|
searches.append(search_name + '=' +
|
||||||
|
saved_searches().lookup(search_name))
|
||||||
|
else:
|
||||||
|
txt = ''
|
||||||
|
else:
|
||||||
|
txt = ''
|
||||||
|
if len(searches) > 1:
|
||||||
|
self.saved_searches_label.setText('\n'.join(searches))
|
||||||
|
else:
|
||||||
|
self.saved_searches_label.setText('')
|
||||||
|
|
||||||
|
def name_text_edited(self, new_name):
|
||||||
|
self.new_name = unicode(new_name)
|
||||||
|
|
||||||
|
def name_index_changed(self, dex):
|
||||||
|
if self.editing and (self.vl_text.text() != self.original_search or
|
||||||
|
self.new_name != self.editing):
|
||||||
|
if not question_dialog(self.gui, _('Search text changed'),
|
||||||
|
_('The virtual library name or the search text has changed. '
|
||||||
|
'Do you want to discard these changes?'),
|
||||||
|
default_yes=False):
|
||||||
|
self.vl_name.blockSignals(True)
|
||||||
|
self.vl_name.setCurrentIndex(self.original_index)
|
||||||
|
self.vl_name.lineEdit().setText(self.new_name)
|
||||||
|
self.vl_name.blockSignals(False)
|
||||||
|
return
|
||||||
|
self.new_name = self.editing = self.vl_name.currentText()
|
||||||
|
self.original_index = dex
|
||||||
|
self.original_search = unicode(self.vl_name.itemData(dex).toString())
|
||||||
|
self.vl_text.setText(self.original_search)
|
||||||
|
|
||||||
|
def link_activated(self, url):
|
||||||
|
db = self.gui.current_db
|
||||||
|
f, txt = unicode(url).partition('.')[0::2]
|
||||||
|
if f == 'search':
|
||||||
|
names = saved_searches().names()
|
||||||
|
else:
|
||||||
|
names = getattr(db, 'all_%s_names'%f)()
|
||||||
|
d = SelectNames(names, txt, parent=self)
|
||||||
|
if d.exec_() == d.Accepted:
|
||||||
|
prefix = f+'s' if f in {'tag', 'author'} else f
|
||||||
|
if f == 'search':
|
||||||
|
search = ['(%s)'%(saved_searches().lookup(x)) for x in d.names]
|
||||||
|
else:
|
||||||
|
search = ['%s:"=%s"'%(prefix, x.replace('"', '\\"')) for x in d.names]
|
||||||
|
if search:
|
||||||
|
if not self.editing:
|
||||||
|
self.vl_name.lineEdit().setText(d.names.next())
|
||||||
|
self.vl_name.lineEdit().setCursorPosition(0)
|
||||||
|
self.vl_text.setText(d.match_type.join(search))
|
||||||
|
self.vl_text.setCursorPosition(0)
|
||||||
|
|
||||||
|
def accept(self):
|
||||||
|
n = unicode(self.vl_name.currentText()).strip()
|
||||||
|
if not n:
|
||||||
|
error_dialog(self.gui, _('No name'),
|
||||||
|
_('You must provide a name for the new virtual library'),
|
||||||
|
show=True)
|
||||||
|
return
|
||||||
|
|
||||||
|
if n.startswith('*'):
|
||||||
|
error_dialog(self.gui, _('Invalid name'),
|
||||||
|
_('A virtual library name cannot begin with "*"'),
|
||||||
|
show=True)
|
||||||
|
return
|
||||||
|
|
||||||
|
if n in self.existing_names and n != self.editing:
|
||||||
|
if not question_dialog(self.gui, _('Name already in use'),
|
||||||
|
_('That name is already in use. Do you want to replace it '
|
||||||
|
'with the new search?'),
|
||||||
|
default_yes=False):
|
||||||
|
return
|
||||||
|
|
||||||
|
v = unicode(self.vl_text.text()).strip()
|
||||||
|
if not v:
|
||||||
|
error_dialog(self.gui, _('No search string'),
|
||||||
|
_('You must provide a search to define the new virtual library'),
|
||||||
|
show=True)
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
db = self.gui.library_view.model().db
|
||||||
|
recs = db.data.search_getting_ids('', v, use_virtual_library=False)
|
||||||
|
except ParseException as e:
|
||||||
|
error_dialog(self.gui, _('Invalid search'),
|
||||||
|
_('The search in the search box is not valid'),
|
||||||
|
det_msg=e.msg, show=True)
|
||||||
|
return
|
||||||
|
|
||||||
|
if not recs and not question_dialog(
|
||||||
|
self.gui, _('Search found no books'),
|
||||||
|
_('The search found no books, so the virtual library '
|
||||||
|
'will be empty. Do you really want to use that search?'),
|
||||||
|
default_yes=False):
|
||||||
|
return
|
||||||
|
|
||||||
|
self.library_name = n
|
||||||
|
self.library_search = v
|
||||||
|
QDialog.accept(self)
|
||||||
|
# }}}
|
||||||
|
|
||||||
class SearchRestrictionMixin(object):
|
class SearchRestrictionMixin(object):
|
||||||
|
|
||||||
|
no_restriction = _('<None>')
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.search_restriction.initialize(help_text=_('Restrict to'))
|
self.checked = QIcon(I('ok.png'))
|
||||||
self.search_restriction.activated[int].connect(self.apply_search_restriction)
|
self.empty = QIcon()
|
||||||
self.library_view.model().count_changed_signal.connect(self.set_number_of_books_shown)
|
self.search_based_vl_name = None
|
||||||
self.search_restriction.setSizeAdjustPolicy(
|
self.search_based_vl = None
|
||||||
self.search_restriction.AdjustToMinimumContentsLengthWithIcon)
|
|
||||||
self.search_restriction.setMinimumContentsLength(10)
|
self.virtual_library_menu = QMenu()
|
||||||
self.search_restriction.setStatusTip(self.search_restriction.toolTip())
|
|
||||||
|
self.virtual_library.clicked.connect(self.virtual_library_clicked)
|
||||||
|
|
||||||
|
self.virtual_library_tooltip = \
|
||||||
|
_('Books display will show only those books matching the search')
|
||||||
|
self.virtual_library.setToolTip(self.virtual_library_tooltip)
|
||||||
|
|
||||||
|
self.search_restriction = ComboBoxWithHelp(self)
|
||||||
|
self.search_restriction.setVisible(False)
|
||||||
self.search_count.setText(_("(all books)"))
|
self.search_count.setText(_("(all books)"))
|
||||||
self.search_restriction_tooltip = \
|
self.ar_menu = QMenu(_('Additional restriction'))
|
||||||
_('Books display will be restricted to those matching a '
|
|
||||||
'selected saved search')
|
def add_virtual_library(self, db, name, search):
|
||||||
self.search_restriction.setToolTip(self.search_restriction_tooltip)
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
virt_libs[name] = search
|
||||||
|
db.prefs.set('virtual_libraries', virt_libs)
|
||||||
|
|
||||||
|
def do_create_edit(self, editing=None):
|
||||||
|
db = self.library_view.model().db
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
cd = CreateVirtualLibrary(self, virt_libs.keys(), editing=editing)
|
||||||
|
if cd.exec_() == cd.Accepted:
|
||||||
|
if editing:
|
||||||
|
self._remove_vl(editing, reapply=False)
|
||||||
|
self.add_virtual_library(db, cd.library_name, cd.library_search)
|
||||||
|
if not editing or editing == db.data.get_base_restriction_name():
|
||||||
|
self.apply_virtual_library(cd.library_name)
|
||||||
|
|
||||||
|
def virtual_library_clicked(self):
|
||||||
|
m = self.virtual_library_menu
|
||||||
|
m.clear()
|
||||||
|
|
||||||
|
a = m.addAction(_('Create Virtual Library'))
|
||||||
|
a.triggered.connect(partial(self.do_create_edit, editing=None))
|
||||||
|
|
||||||
|
self.edit_menu = a = QMenu()
|
||||||
|
a.setTitle(_('Edit Virtual Library'))
|
||||||
|
a.aboutToShow.connect(partial(self.build_virtual_library_list, remove=False))
|
||||||
|
m.addMenu(a)
|
||||||
|
|
||||||
|
self.rm_menu = a = QMenu()
|
||||||
|
a.setTitle(_('Remove Virtual Library'))
|
||||||
|
a.aboutToShow.connect(partial(self.build_virtual_library_list, remove=True))
|
||||||
|
m.addMenu(a)
|
||||||
|
|
||||||
|
m.addSeparator()
|
||||||
|
|
||||||
|
db = self.library_view.model().db
|
||||||
|
|
||||||
|
a = self.ar_menu
|
||||||
|
a.clear()
|
||||||
|
a.setIcon(self.checked if db.data.get_search_restriction_name() else self.empty)
|
||||||
|
a.aboutToShow.connect(self.build_search_restriction_list)
|
||||||
|
m.addMenu(a)
|
||||||
|
|
||||||
|
m.addSeparator()
|
||||||
|
|
||||||
|
current_lib = db.data.get_base_restriction_name()
|
||||||
|
|
||||||
|
if current_lib == '':
|
||||||
|
a = m.addAction(self.checked, self.no_restriction)
|
||||||
|
else:
|
||||||
|
a = m.addAction(self.empty, self.no_restriction)
|
||||||
|
a.triggered.connect(partial(self.apply_virtual_library, library=''))
|
||||||
|
|
||||||
|
a = m.addAction(self.empty, _('*current search'))
|
||||||
|
a.triggered.connect(partial(self.apply_virtual_library, library='*'))
|
||||||
|
|
||||||
|
if self.search_based_vl_name:
|
||||||
|
a = m.addAction(
|
||||||
|
self.checked if db.data.get_base_restriction_name().startswith('*')
|
||||||
|
else self.empty,
|
||||||
|
self.search_based_vl_name)
|
||||||
|
a.triggered.connect(partial(self.apply_virtual_library,
|
||||||
|
library=self.search_based_vl_name))
|
||||||
|
|
||||||
|
m.addSeparator()
|
||||||
|
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
for vl in sorted(virt_libs.keys(), key=sort_key):
|
||||||
|
a = m.addAction(self.checked if vl == current_lib else self.empty, vl)
|
||||||
|
a.triggered.connect(partial(self.apply_virtual_library, library=vl))
|
||||||
|
|
||||||
|
p = QPoint(0, self.virtual_library.height())
|
||||||
|
self.virtual_library_menu.popup(self.virtual_library.mapToGlobal(p))
|
||||||
|
|
||||||
|
def apply_virtual_library(self, library=None):
|
||||||
|
db = self.library_view.model().db
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
if not library:
|
||||||
|
db.data.set_base_restriction('')
|
||||||
|
db.data.set_base_restriction_name('')
|
||||||
|
elif library == '*':
|
||||||
|
if not self.search.current_text:
|
||||||
|
error_dialog(self, _('No search'),
|
||||||
|
_('There is no current search to use'), show=True)
|
||||||
|
return
|
||||||
|
|
||||||
|
txt = _build_full_search_string(self)
|
||||||
|
try:
|
||||||
|
db.data.search_getting_ids('', txt, use_virtual_library=False)
|
||||||
|
except ParseException as e:
|
||||||
|
error_dialog(self, _('Invalid search'),
|
||||||
|
_('The search in the search box is not valid'),
|
||||||
|
det_msg=e.msg, show=True)
|
||||||
|
return
|
||||||
|
|
||||||
|
self.search_based_vl = txt
|
||||||
|
db.data.set_base_restriction(txt)
|
||||||
|
self.search_based_vl_name = self._trim_restriction_name('*' + txt)
|
||||||
|
db.data.set_base_restriction_name(self.search_based_vl_name)
|
||||||
|
elif library == self.search_based_vl_name:
|
||||||
|
db.data.set_base_restriction(self.search_based_vl)
|
||||||
|
db.data.set_base_restriction_name(self.search_based_vl_name)
|
||||||
|
elif library in virt_libs:
|
||||||
|
db.data.set_base_restriction(virt_libs[library])
|
||||||
|
db.data.set_base_restriction_name(library)
|
||||||
|
self.virtual_library.setToolTip(self.virtual_library_tooltip + '\n' +
|
||||||
|
db.data.get_base_restriction())
|
||||||
|
self._apply_search_restriction(db.data.get_search_restriction(),
|
||||||
|
db.data.get_search_restriction_name())
|
||||||
|
|
||||||
|
def build_virtual_library_list(self, remove=False):
|
||||||
|
db = self.library_view.model().db
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
if remove:
|
||||||
|
m = self.rm_menu
|
||||||
|
else:
|
||||||
|
m = self.edit_menu
|
||||||
|
m.clear()
|
||||||
|
|
||||||
|
def add_action(name, search):
|
||||||
|
a = m.addAction(name)
|
||||||
|
if remove:
|
||||||
|
a.triggered.connect(partial(self.remove_vl_triggered, name=name))
|
||||||
|
else:
|
||||||
|
a.triggered.connect(partial(self.do_create_edit, editing=name))
|
||||||
|
|
||||||
|
for n in sorted(virt_libs.keys(), key=sort_key):
|
||||||
|
add_action(n, virt_libs[n])
|
||||||
|
|
||||||
|
def remove_vl_triggered(self, name=None):
|
||||||
|
if not question_dialog(self, _('Are you sure?'),
|
||||||
|
_('Are you sure you want to remove '
|
||||||
|
'the virtual library {0}').format(name),
|
||||||
|
default_yes=False):
|
||||||
|
return
|
||||||
|
self._remove_vl(name, reapply=True)
|
||||||
|
|
||||||
|
def _remove_vl(self, name, reapply=True):
|
||||||
|
db = self.library_view.model().db
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
|
virt_libs.pop(name, None)
|
||||||
|
db.prefs.set('virtual_libraries', virt_libs)
|
||||||
|
if reapply and db.data.get_base_restriction_name() == name:
|
||||||
|
self.apply_virtual_library('')
|
||||||
|
|
||||||
|
def _trim_restriction_name(self, name):
|
||||||
|
return name[0:MAX_VIRTUAL_LIBRARY_NAME_LENGTH].strip()
|
||||||
|
|
||||||
|
def build_search_restriction_list(self):
|
||||||
|
m = self.ar_menu
|
||||||
|
m.clear()
|
||||||
|
|
||||||
|
current_restriction_text = None
|
||||||
|
|
||||||
|
if self.search_restriction.count() > 1:
|
||||||
|
txt = unicode(self.search_restriction.itemText(2))
|
||||||
|
if txt.startswith('*'):
|
||||||
|
current_restriction_text = txt
|
||||||
|
self.search_restriction.clear()
|
||||||
|
|
||||||
|
current_restriction = self.library_view.model().db.data.get_search_restriction_name()
|
||||||
|
m.setIcon(self.checked if current_restriction else self.empty)
|
||||||
|
|
||||||
|
def add_action(txt, index):
|
||||||
|
self.search_restriction.addItem(txt)
|
||||||
|
txt = self._trim_restriction_name(txt)
|
||||||
|
if txt == current_restriction:
|
||||||
|
a = m.addAction(self.checked, txt if txt else self.no_restriction)
|
||||||
|
else:
|
||||||
|
a = m.addAction(self.empty, txt if txt else self.no_restriction)
|
||||||
|
a.triggered.connect(partial(self.search_restriction_triggered,
|
||||||
|
action=a, index=index))
|
||||||
|
|
||||||
|
add_action('', 0)
|
||||||
|
add_action(_('*current search'), 1)
|
||||||
|
dex = 2
|
||||||
|
if current_restriction_text:
|
||||||
|
add_action(current_restriction_text, 2)
|
||||||
|
dex += 1
|
||||||
|
|
||||||
|
for n in sorted(saved_searches().names(), key=sort_key):
|
||||||
|
add_action(n, dex)
|
||||||
|
dex += 1
|
||||||
|
|
||||||
|
def search_restriction_triggered(self, action=None, index=None):
|
||||||
|
self.search_restriction.setCurrentIndex(index)
|
||||||
|
self.apply_search_restriction(index)
|
||||||
|
|
||||||
def apply_named_search_restriction(self, name):
|
def apply_named_search_restriction(self, name):
|
||||||
if not name:
|
if not name:
|
||||||
@ -29,15 +511,14 @@ class SearchRestrictionMixin(object):
|
|||||||
r = self.search_restriction.findText(name)
|
r = self.search_restriction.findText(name)
|
||||||
if r < 0:
|
if r < 0:
|
||||||
r = 0
|
r = 0
|
||||||
if r != self.search_restriction.currentIndex():
|
self.search_restriction.setCurrentIndex(r)
|
||||||
self.search_restriction.setCurrentIndex(r)
|
self.apply_search_restriction(r)
|
||||||
self.apply_search_restriction(r)
|
|
||||||
|
|
||||||
def apply_text_search_restriction(self, search):
|
def apply_text_search_restriction(self, search):
|
||||||
search = unicode(search)
|
search = unicode(search)
|
||||||
if not search:
|
if not search:
|
||||||
self.search_restriction.setCurrentIndex(0)
|
self.search_restriction.setCurrentIndex(0)
|
||||||
self._apply_search_restriction('')
|
self._apply_search_restriction('', '')
|
||||||
else:
|
else:
|
||||||
s = '*' + search
|
s = '*' + search
|
||||||
if self.search_restriction.count() > 1:
|
if self.search_restriction.count() > 1:
|
||||||
@ -49,10 +530,7 @@ class SearchRestrictionMixin(object):
|
|||||||
else:
|
else:
|
||||||
self.search_restriction.insertItem(2, s)
|
self.search_restriction.insertItem(2, s)
|
||||||
self.search_restriction.setCurrentIndex(2)
|
self.search_restriction.setCurrentIndex(2)
|
||||||
self.search_restriction.setToolTip('<p>' +
|
self._apply_search_restriction(search, self._trim_restriction_name(s))
|
||||||
self.search_restriction_tooltip +
|
|
||||||
_(' or the search ') + "'" + search + "'</p>")
|
|
||||||
self._apply_search_restriction(search)
|
|
||||||
|
|
||||||
def apply_search_restriction(self, i):
|
def apply_search_restriction(self, i):
|
||||||
if i == 1:
|
if i == 1:
|
||||||
@ -66,18 +544,20 @@ class SearchRestrictionMixin(object):
|
|||||||
restriction = 'search:"%s"'%(r)
|
restriction = 'search:"%s"'%(r)
|
||||||
else:
|
else:
|
||||||
restriction = ''
|
restriction = ''
|
||||||
self._apply_search_restriction(restriction)
|
self._apply_search_restriction(restriction, r)
|
||||||
|
|
||||||
def _apply_search_restriction(self, restriction):
|
def _apply_search_restriction(self, restriction, name):
|
||||||
self.saved_search.clear()
|
self.saved_search.clear()
|
||||||
# The order below is important. Set the restriction, force a '' search
|
# The order below is important. Set the restriction, force a '' search
|
||||||
# to apply it, reset the tag browser to take it into account, then set
|
# to apply it, reset the tag browser to take it into account, then set
|
||||||
# the book count.
|
# the book count.
|
||||||
self.library_view.model().db.data.set_search_restriction(restriction)
|
self.library_view.model().db.data.set_search_restriction(restriction)
|
||||||
|
self.library_view.model().db.data.set_search_restriction_name(name)
|
||||||
self.search.clear(emit_search=True)
|
self.search.clear(emit_search=True)
|
||||||
self.tags_view.set_search_restriction(restriction)
|
self.tags_view.recount()
|
||||||
self.set_number_of_books_shown()
|
self.set_number_of_books_shown()
|
||||||
self.current_view().setFocus(Qt.OtherFocusReason)
|
self.current_view().setFocus(Qt.OtherFocusReason)
|
||||||
|
self.set_window_title()
|
||||||
|
|
||||||
def set_number_of_books_shown(self):
|
def set_number_of_books_shown(self):
|
||||||
db = self.library_view.model().db
|
db = self.library_view.model().db
|
||||||
@ -86,9 +566,9 @@ class SearchRestrictionMixin(object):
|
|||||||
rows = self.current_view().row_count()
|
rows = self.current_view().row_count()
|
||||||
rbc = max(rows, db.data.get_search_restriction_book_count())
|
rbc = max(rows, db.data.get_search_restriction_book_count())
|
||||||
t = _("({0} of {1})").format(rows, rbc)
|
t = _("({0} of {1})").format(rows, rbc)
|
||||||
self.search_count.setStyleSheet \
|
self.search_count.setStyleSheet(
|
||||||
('QLabel { border-radius: 8px; background-color: yellow; }')
|
'QLabel { border-radius: 8px; background-color: yellow; }')
|
||||||
else: # No restriction or not library view
|
else: # No restriction or not library view
|
||||||
if not self.search.in_a_search():
|
if not self.search.in_a_search():
|
||||||
t = _("(all books)")
|
t = _("(all books)")
|
||||||
else:
|
else:
|
||||||
@ -96,3 +576,14 @@ class SearchRestrictionMixin(object):
|
|||||||
self.search_count.setStyleSheet(
|
self.search_count.setStyleSheet(
|
||||||
'QLabel { background-color: transparent; }')
|
'QLabel { background-color: transparent; }')
|
||||||
self.search_count.setText(t)
|
self.search_count.setText(t)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
from calibre.gui2 import Application
|
||||||
|
from calibre.gui2.preferences import init_gui
|
||||||
|
app = Application([])
|
||||||
|
app
|
||||||
|
gui = init_gui()
|
||||||
|
d = CreateVirtualLibrary(gui, [])
|
||||||
|
d.exec_()
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,91 +1,104 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
from __future__ import (unicode_literals, division, absolute_import, print_function)
|
from __future__ import (unicode_literals, division, absolute_import, print_function)
|
||||||
store_version = 2 # Needed for dynamic plugin loading
|
store_version = 3 # Needed for dynamic plugin loading
|
||||||
|
|
||||||
__license__ = 'GPL 3'
|
__license__ = 'GPL 3'
|
||||||
__copyright__ = '2011, John Schember <john@nachtimwald.com>'
|
__copyright__ = '2011, 2013, John Schember <john@nachtimwald.com>'
|
||||||
__docformat__ = 'restructuredtext en'
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
|
import base64
|
||||||
import mimetypes
|
import mimetypes
|
||||||
|
import re
|
||||||
import urllib
|
import urllib
|
||||||
from contextlib import closing
|
from contextlib import closing
|
||||||
|
|
||||||
from lxml import html
|
from lxml import etree
|
||||||
|
|
||||||
from PyQt4.Qt import QUrl
|
from calibre import browser, url_slash_cleaner
|
||||||
|
from calibre.constants import __version__
|
||||||
from calibre import browser, random_user_agent, 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.basic_config import BasicStoreConfig
|
||||||
|
from calibre.gui2.store.opensearch_store import OpenSearchOPDSStore
|
||||||
from calibre.gui2.store.search_result import SearchResult
|
from calibre.gui2.store.search_result import SearchResult
|
||||||
from calibre.gui2.store.web_store_dialog import WebStoreDialog
|
|
||||||
|
|
||||||
class GutenbergStore(BasicStoreConfig, StorePlugin):
|
class GutenbergStore(BasicStoreConfig, OpenSearchOPDSStore):
|
||||||
|
|
||||||
def open(self, parent=None, detail_item=None, external=False):
|
open_search_url = 'http://www.gutenberg.org/catalog/osd-books.xml'
|
||||||
url = 'http://gutenberg.org/'
|
web_url = 'http://m.gutenberg.org/'
|
||||||
|
|
||||||
if detail_item:
|
|
||||||
detail_item = url_slash_cleaner(url + detail_item)
|
|
||||||
|
|
||||||
if external or self.config.get('open_external', False):
|
|
||||||
open_url(QUrl(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):
|
def search(self, query, max_results=10, timeout=60):
|
||||||
url = 'http://m.gutenberg.org/ebooks/search.mobile/?default_prefix=all&sort_order=title&query=' + urllib.quote_plus(query)
|
'''
|
||||||
|
Gutenberg's ODPS feed is poorly implmented and has a number of issues
|
||||||
|
which require very special handling to fix the results.
|
||||||
|
|
||||||
br = browser(user_agent=random_user_agent())
|
Issues:
|
||||||
|
* "Sort Alphabetically" and "Sort by Release Date" are returned
|
||||||
|
as book entries.
|
||||||
|
* The author is put into a "content" tag and not the author tag.
|
||||||
|
* The link to the book itself goes to an odps page which we need
|
||||||
|
to turn into a link to a web page.
|
||||||
|
* acquisition links are not part of the search result so we have
|
||||||
|
to go to the odps item itself. Detail item pages have a nasty
|
||||||
|
note saying:
|
||||||
|
DON'T USE THIS PAGE FOR SCRAPING.
|
||||||
|
Seriously. You'll only get your IP blocked.
|
||||||
|
We're using the ODPS feed because people are getting blocked with
|
||||||
|
the previous implementation so due to this using ODPS probably
|
||||||
|
won't solve this issue.
|
||||||
|
* Images are not links but base64 encoded strings. They are also not
|
||||||
|
real cover images but a little blue book thumbnail.
|
||||||
|
'''
|
||||||
|
|
||||||
|
url = 'http://m.gutenberg.org/ebooks/search.opds/?query=' + urllib.quote_plus(query)
|
||||||
|
|
||||||
counter = max_results
|
counter = max_results
|
||||||
|
br = browser(user_agent='calibre/'+__version__)
|
||||||
with closing(br.open(url, timeout=timeout)) as f:
|
with closing(br.open(url, timeout=timeout)) as f:
|
||||||
doc = html.fromstring(f.read())
|
doc = etree.fromstring(f.read())
|
||||||
for data in doc.xpath('//ol[@class="results"]/li[@class="booklink"]'):
|
for data in doc.xpath('//*[local-name() = "entry"]'):
|
||||||
if counter <= 0:
|
if counter <= 0:
|
||||||
break
|
break
|
||||||
|
|
||||||
id = ''.join(data.xpath('./a/@href'))
|
|
||||||
id = id.split('.mobile')[0]
|
|
||||||
|
|
||||||
title = ''.join(data.xpath('.//span[@class="title"]/text()'))
|
|
||||||
author = ''.join(data.xpath('.//span[@class="subtitle"]/text()'))
|
|
||||||
|
|
||||||
counter -= 1
|
counter -= 1
|
||||||
|
|
||||||
s = SearchResult()
|
s = SearchResult()
|
||||||
s.cover_url = ''
|
|
||||||
|
|
||||||
s.detail_item = id.strip()
|
# We could use the <link rel="alternate" type="text/html" ...> tag from the
|
||||||
s.title = title.strip()
|
# detail odps page but this is easier.
|
||||||
s.author = author.strip()
|
id = ''.join(data.xpath('./*[local-name() = "id"]/text()')).strip()
|
||||||
s.price = '$0.00'
|
s.detail_item = url_slash_cleaner('%s/ebooks/%s' % (self.web_url, re.sub('[^\d]', '', id)))
|
||||||
s.drm = SearchResult.DRM_UNLOCKED
|
if not s.detail_item:
|
||||||
|
continue
|
||||||
|
|
||||||
|
s.title = ' '.join(data.xpath('./*[local-name() = "title"]//text()')).strip()
|
||||||
|
s.author = ', '.join(data.xpath('./*[local-name() = "content"]//text()')).strip()
|
||||||
|
if not s.title or not s.author:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Get the formats and direct download links.
|
||||||
|
with closing(br.open(id, timeout=timeout/4)) as nf:
|
||||||
|
ndoc = etree.fromstring(nf.read())
|
||||||
|
for link in ndoc.xpath('//*[local-name() = "link" and @rel = "http://opds-spec.org/acquisition"]'):
|
||||||
|
type = link.get('type')
|
||||||
|
href = link.get('href')
|
||||||
|
if type:
|
||||||
|
ext = mimetypes.guess_extension(type)
|
||||||
|
if ext:
|
||||||
|
ext = ext[1:].upper().strip()
|
||||||
|
s.downloads[ext] = href
|
||||||
|
|
||||||
|
s.formats = ', '.join(s.downloads.keys())
|
||||||
|
if not s.formats:
|
||||||
|
continue
|
||||||
|
|
||||||
|
for link in data.xpath('./*[local-name() = "link"]'):
|
||||||
|
rel = link.get('rel')
|
||||||
|
href = link.get('href')
|
||||||
|
type = link.get('type')
|
||||||
|
|
||||||
|
if rel and href and type:
|
||||||
|
if rel in ('http://opds-spec.org/thumbnail', 'http://opds-spec.org/image/thumbnail'):
|
||||||
|
if href.startswith('data:image/png;base64,'):
|
||||||
|
s.cover_data = base64.b64decode(href.replace('data:image/png;base64,', ''))
|
||||||
|
|
||||||
yield s
|
yield s
|
||||||
|
|
||||||
def get_details(self, search_result, timeout):
|
|
||||||
url = url_slash_cleaner('http://m.gutenberg.org/' + search_result.detail_item)
|
|
||||||
|
|
||||||
br = browser(user_agent=random_user_agent())
|
|
||||||
with closing(br.open(url, timeout=timeout)) as nf:
|
|
||||||
doc = html.fromstring(nf.read())
|
|
||||||
|
|
||||||
for save_item in doc.xpath('//li[contains(@class, "icon_save")]/a'):
|
|
||||||
type = save_item.get('type')
|
|
||||||
href = save_item.get('href')
|
|
||||||
|
|
||||||
if type:
|
|
||||||
ext = mimetypes.guess_extension(type)
|
|
||||||
if ext:
|
|
||||||
ext = ext[1:].upper().strip()
|
|
||||||
search_result.downloads[ext] = href
|
|
||||||
|
|
||||||
search_result.formats = ', '.join(search_result.downloads.keys())
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
82
src/calibre/gui2/store/stores/koobe_plugin.py
Normal file
82
src/calibre/gui2/store/stores/koobe_plugin.py
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from __future__ import (division, absolute_import, print_function)
|
||||||
|
store_version = 1 # Needed for dynamic plugin loading
|
||||||
|
|
||||||
|
__license__ = 'GPL 3'
|
||||||
|
__copyright__ = '2013, Tomasz Długosz <tomek3d@gmail.com>'
|
||||||
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
|
import urllib
|
||||||
|
from base64 import b64encode
|
||||||
|
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 KoobeStore(BasicStoreConfig, StorePlugin):
|
||||||
|
|
||||||
|
def open(self, parent=None, detail_item=None, external=False):
|
||||||
|
aff_root = 'https://www.a4b-tracking.com/pl/stat-click-text-link/15/58/'
|
||||||
|
url = 'http://www.koobe.pl/'
|
||||||
|
|
||||||
|
aff_url = aff_root + str(b64encode(url))
|
||||||
|
|
||||||
|
detail_url = None
|
||||||
|
if detail_item:
|
||||||
|
detail_url = aff_root + str(b64encode(detail_item))
|
||||||
|
|
||||||
|
if external or self.config.get('open_external', False):
|
||||||
|
open_url(QUrl(url_slash_cleaner(detail_url if detail_url else aff_url)))
|
||||||
|
else:
|
||||||
|
d = WebStoreDialog(self.gui, url, parent, detail_url if detail_url else aff_url)
|
||||||
|
d.setWindowTitle(self.name)
|
||||||
|
d.set_tags(self.config.get('tags', ''))
|
||||||
|
d.exec_()
|
||||||
|
|
||||||
|
def search(self, query, max_results=10, timeout=60):
|
||||||
|
|
||||||
|
br = browser()
|
||||||
|
page=1
|
||||||
|
|
||||||
|
counter = max_results
|
||||||
|
while counter:
|
||||||
|
with closing(br.open('http://www.koobe.pl/s,p,' + str(page) + ',szukaj/fraza:' + urllib.quote(query), timeout=timeout)) as f:
|
||||||
|
doc = html.fromstring(f.read().decode('utf-8'))
|
||||||
|
for data in doc.xpath('//div[@class="seach_result"]/div[@class="result"]'):
|
||||||
|
if counter <= 0:
|
||||||
|
break
|
||||||
|
|
||||||
|
id = ''.join(data.xpath('.//div[@class="cover"]/a/@href'))
|
||||||
|
if not id:
|
||||||
|
continue
|
||||||
|
|
||||||
|
cover_url = ''.join(data.xpath('.//div[@class="cover"]/a/img/@src'))
|
||||||
|
price = ''.join(data.xpath('.//span[@class="current_price"]/text()'))
|
||||||
|
title = ''.join(data.xpath('.//h2[@class="title"]/a/text()'))
|
||||||
|
author = ''.join(data.xpath('.//h3[@class="book_author"]/a/text()'))
|
||||||
|
formats = ', '.join(data.xpath('.//div[@class="formats"]/div/div/@title'))
|
||||||
|
|
||||||
|
counter -= 1
|
||||||
|
|
||||||
|
s = SearchResult()
|
||||||
|
s.cover_url = 'http://koobe.pl/' + cover_url
|
||||||
|
s.title = title.strip()
|
||||||
|
s.author = author.strip()
|
||||||
|
s.price = price
|
||||||
|
s.detail_item = 'http://koobe.pl' + id[1:]
|
||||||
|
s.formats = formats.upper()
|
||||||
|
s.drm = SearchResult.DRM_UNLOCKED
|
||||||
|
|
||||||
|
yield s
|
||||||
|
if not doc.xpath('//div[@class="site_bottom"]//a[@class="right"]'):
|
||||||
|
break
|
||||||
|
page+=1
|
@ -1,14 +1,15 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
from __future__ import (unicode_literals, division, absolute_import, print_function)
|
from __future__ import (unicode_literals, division, absolute_import, print_function)
|
||||||
store_version = 1 # Needed for dynamic plugin loading
|
store_version = 2 # Needed for dynamic plugin loading
|
||||||
|
|
||||||
__license__ = 'GPL 3'
|
__license__ = 'GPL 3'
|
||||||
__copyright__ = '2011-2012, Tomasz Długosz <tomek3d@gmail.com>'
|
__copyright__ = '2011-2013, Tomasz Długosz <tomek3d@gmail.com>'
|
||||||
__docformat__ = 'restructuredtext en'
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|
||||||
import re
|
import re
|
||||||
import urllib
|
import urllib
|
||||||
|
from base64 import b64encode
|
||||||
from contextlib import closing
|
from contextlib import closing
|
||||||
|
|
||||||
from lxml import html
|
from lxml import html
|
||||||
@ -25,17 +26,19 @@ from calibre.gui2.store.web_store_dialog import WebStoreDialog
|
|||||||
class WoblinkStore(BasicStoreConfig, StorePlugin):
|
class WoblinkStore(BasicStoreConfig, StorePlugin):
|
||||||
|
|
||||||
def open(self, parent=None, detail_item=None, external=False):
|
def open(self, parent=None, detail_item=None, external=False):
|
||||||
|
aff_root = 'https://www.a4b-tracking.com/pl/stat-click-text-link/16/58/'
|
||||||
url = 'http://woblink.com/publication'
|
url = 'http://woblink.com/publication'
|
||||||
|
|
||||||
|
aff_url = aff_root + str(b64encode(url))
|
||||||
detail_url = None
|
detail_url = None
|
||||||
|
|
||||||
if detail_item:
|
if detail_item:
|
||||||
detail_url = 'http://woblink.com' + detail_item
|
detail_url = aff_root + str(b64encode('http://woblink.com' + detail_item))
|
||||||
|
|
||||||
if external or self.config.get('open_external', False):
|
if external or self.config.get('open_external', False):
|
||||||
open_url(QUrl(url_slash_cleaner(detail_url if detail_url else url)))
|
open_url(QUrl(url_slash_cleaner(detail_url if detail_url else aff_url)))
|
||||||
else:
|
else:
|
||||||
d = WebStoreDialog(self.gui, url, parent, detail_url)
|
d = WebStoreDialog(self.gui, url, parent, detail_url if detail_url else aff_url)
|
||||||
d.setWindowTitle(self.name)
|
d.setWindowTitle(self.name)
|
||||||
d.set_tags(self.config.get('tags', ''))
|
d.set_tags(self.config.get('tags', ''))
|
||||||
d.exec_()
|
d.exec_()
|
||||||
|
@ -264,13 +264,8 @@ class TagsModel(QAbstractItemModel): # {{{
|
|||||||
if rebuild:
|
if rebuild:
|
||||||
self.rebuild_node_tree(state_map)
|
self.rebuild_node_tree(state_map)
|
||||||
|
|
||||||
def set_search_restriction(self, s):
|
|
||||||
self.search_restriction = s
|
|
||||||
self.rebuild_node_tree()
|
|
||||||
|
|
||||||
def set_database(self, db):
|
def set_database(self, db):
|
||||||
self.beginResetModel()
|
self.beginResetModel()
|
||||||
self.search_restriction = None
|
|
||||||
hidden_cats = db.prefs.get('tag_browser_hidden_categories', None)
|
hidden_cats = db.prefs.get('tag_browser_hidden_categories', None)
|
||||||
# migrate from config to db prefs
|
# migrate from config to db prefs
|
||||||
if hidden_cats is None:
|
if hidden_cats is None:
|
||||||
@ -848,7 +843,7 @@ class TagsModel(QAbstractItemModel): # {{{
|
|||||||
self.categories = {}
|
self.categories = {}
|
||||||
|
|
||||||
# Get the categories
|
# Get the categories
|
||||||
if self.search_restriction:
|
if self.db.data.get_base_restriction() or self.db.data.get_search_restriction():
|
||||||
try:
|
try:
|
||||||
data = self.db.get_categories(sort=sort,
|
data = self.db.get_categories(sort=sort,
|
||||||
icon_map=self.category_icon_map,
|
icon_map=self.category_icon_map,
|
||||||
|
@ -232,10 +232,6 @@ class TagsView(QTreeView): # {{{
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def set_search_restriction(self, s):
|
|
||||||
s = s if s else None
|
|
||||||
self._model.set_search_restriction(s)
|
|
||||||
|
|
||||||
def mouseMoveEvent(self, event):
|
def mouseMoveEvent(self, event):
|
||||||
dex = self.indexAt(event.pos())
|
dex = self.indexAt(event.pos())
|
||||||
if self.in_drag_drop or not dex.isValid():
|
if self.in_drag_drop or not dex.isValid():
|
||||||
|
@ -11,14 +11,14 @@ from base64 import b64encode
|
|||||||
|
|
||||||
from PyQt4.Qt import (QWidget, QGridLayout, QListWidget, QSize, Qt, QUrl,
|
from PyQt4.Qt import (QWidget, QGridLayout, QListWidget, QSize, Qt, QUrl,
|
||||||
pyqtSlot, pyqtSignal, QVBoxLayout, QFrame, QLabel,
|
pyqtSlot, pyqtSignal, QVBoxLayout, QFrame, QLabel,
|
||||||
QLineEdit, QTimer, QPushButton, QIcon)
|
QLineEdit, QTimer, QPushButton, QIcon, QSplitter)
|
||||||
from PyQt4.QtWebKit import QWebView, QWebPage, QWebElement
|
from PyQt4.QtWebKit import QWebView, QWebPage, QWebElement
|
||||||
|
|
||||||
from calibre.ebooks.oeb.display.webview import load_html
|
from calibre.ebooks.oeb.display.webview import load_html
|
||||||
from calibre.gui2 import error_dialog, question_dialog
|
from calibre.gui2 import error_dialog, question_dialog, gprefs
|
||||||
from calibre.utils.logging import default_log
|
from calibre.utils.logging import default_log
|
||||||
|
|
||||||
class Page(QWebPage): # {{{
|
class Page(QWebPage): # {{{
|
||||||
|
|
||||||
elem_clicked = pyqtSignal(object, object, object, object)
|
elem_clicked = pyqtSignal(object, object, object, object)
|
||||||
|
|
||||||
@ -67,7 +67,7 @@ class Page(QWebPage): # {{{
|
|||||||
self.evaljs(self.js)
|
self.evaljs(self.js)
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class WebView(QWebView): # {{{
|
class WebView(QWebView): # {{{
|
||||||
|
|
||||||
elem_clicked = pyqtSignal(object, object, object, object)
|
elem_clicked = pyqtSignal(object, object, object, object)
|
||||||
|
|
||||||
@ -106,38 +106,46 @@ class ItemEdit(QWidget):
|
|||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
QWidget.__init__(self, parent)
|
QWidget.__init__(self, parent)
|
||||||
self.l = l = QGridLayout()
|
self.setLayout(QVBoxLayout())
|
||||||
self.setLayout(l)
|
|
||||||
|
|
||||||
self.la = la = QLabel('<b>'+_(
|
self.la = la = QLabel('<b>'+_(
|
||||||
'Select a destination for the Table of Contents entry'))
|
'Select a destination for the Table of Contents entry'))
|
||||||
l.addWidget(la, 0, 0, 1, 3)
|
self.layout().addWidget(la)
|
||||||
|
self.splitter = sp = QSplitter(self)
|
||||||
|
self.layout().addWidget(sp)
|
||||||
|
self.layout().setStretch(1, 10)
|
||||||
|
sp.setOpaqueResize(False)
|
||||||
|
sp.setChildrenCollapsible(False)
|
||||||
|
|
||||||
self.dest_list = dl = QListWidget(self)
|
self.dest_list = dl = QListWidget(self)
|
||||||
dl.setMinimumWidth(250)
|
dl.setMinimumWidth(250)
|
||||||
dl.currentItemChanged.connect(self.current_changed)
|
dl.currentItemChanged.connect(self.current_changed)
|
||||||
l.addWidget(dl, 1, 0, 2, 1)
|
sp.addWidget(dl)
|
||||||
|
|
||||||
|
w = self.w = QWidget(self)
|
||||||
|
l = w.l = QGridLayout()
|
||||||
|
w.setLayout(l)
|
||||||
self.view = WebView(self)
|
self.view = WebView(self)
|
||||||
self.view.elem_clicked.connect(self.elem_clicked)
|
self.view.elem_clicked.connect(self.elem_clicked)
|
||||||
l.addWidget(self.view, 1, 1, 1, 3)
|
l.addWidget(self.view, 0, 0, 1, 3)
|
||||||
|
sp.addWidget(w)
|
||||||
|
|
||||||
|
self.search_text = s = QLineEdit(self)
|
||||||
|
s.setPlaceholderText(_('Search for text...'))
|
||||||
|
l.addWidget(s, 1, 0)
|
||||||
|
self.ns_button = b = QPushButton(QIcon(I('arrow-down.png')), _('Find &next'), self)
|
||||||
|
b.clicked.connect(self.find_next)
|
||||||
|
l.addWidget(b, 1, 1)
|
||||||
|
self.ps_button = b = QPushButton(QIcon(I('arrow-up.png')), _('Find &previous'), self)
|
||||||
|
l.addWidget(b, 1, 2)
|
||||||
|
b.clicked.connect(self.find_previous)
|
||||||
|
|
||||||
self.f = f = QFrame()
|
self.f = f = QFrame()
|
||||||
f.setFrameShape(f.StyledPanel)
|
f.setFrameShape(f.StyledPanel)
|
||||||
f.setMinimumWidth(250)
|
f.setMinimumWidth(250)
|
||||||
l.addWidget(f, 1, 4, 2, 1)
|
|
||||||
self.search_text = s = QLineEdit(self)
|
|
||||||
s.setPlaceholderText(_('Search for text...'))
|
|
||||||
l.addWidget(s, 2, 1, 1, 1)
|
|
||||||
self.ns_button = b = QPushButton(QIcon(I('arrow-down.png')), _('Find &next'), self)
|
|
||||||
b.clicked.connect(self.find_next)
|
|
||||||
l.addWidget(b, 2, 2, 1, 1)
|
|
||||||
self.ps_button = b = QPushButton(QIcon(I('arrow-up.png')), _('Find &previous'), self)
|
|
||||||
l.addWidget(b, 2, 3, 1, 1)
|
|
||||||
b.clicked.connect(self.find_previous)
|
|
||||||
l.setRowStretch(1, 10)
|
|
||||||
l = f.l = QVBoxLayout()
|
l = f.l = QVBoxLayout()
|
||||||
f.setLayout(l)
|
f.setLayout(l)
|
||||||
|
sp.addWidget(f)
|
||||||
|
|
||||||
f.la = la = QLabel('<p>'+_(
|
f.la = la = QLabel('<p>'+_(
|
||||||
'Here you can choose a destination for the Table of Contents\' entry'
|
'Here you can choose a destination for the Table of Contents\' entry'
|
||||||
@ -167,6 +175,10 @@ class ItemEdit(QWidget):
|
|||||||
|
|
||||||
l.addStretch()
|
l.addStretch()
|
||||||
|
|
||||||
|
state = gprefs.get('toc_edit_splitter_state', None)
|
||||||
|
if state is not None:
|
||||||
|
sp.restoreState(state)
|
||||||
|
|
||||||
def keyPressEvent(self, ev):
|
def keyPressEvent(self, ev):
|
||||||
if ev.key() in (Qt.Key_Return, Qt.Key_Enter) and self.search_text.hasFocus():
|
if ev.key() in (Qt.Key_Return, Qt.Key_Enter) and self.search_text.hasFocus():
|
||||||
# Prevent pressing enter in the search box from triggering the dialog's accept() method
|
# Prevent pressing enter in the search box from triggering the dialog's accept() method
|
||||||
@ -236,6 +248,7 @@ class ItemEdit(QWidget):
|
|||||||
if item is not None:
|
if item is not None:
|
||||||
if where is None:
|
if where is None:
|
||||||
self.name.setText(item.data(0, Qt.DisplayRole).toString())
|
self.name.setText(item.data(0, Qt.DisplayRole).toString())
|
||||||
|
self.name.setCursorPosition(0)
|
||||||
toc = item.data(0, Qt.UserRole).toPyObject()
|
toc = item.data(0, Qt.UserRole).toPyObject()
|
||||||
if toc.dest:
|
if toc.dest:
|
||||||
for i in xrange(self.dest_list.count()):
|
for i in xrange(self.dest_list.count()):
|
||||||
@ -272,7 +285,6 @@ class ItemEdit(QWidget):
|
|||||||
loctext = _('Approximately %d%% from the top')%frac
|
loctext = _('Approximately %d%% from the top')%frac
|
||||||
return loctext
|
return loctext
|
||||||
|
|
||||||
|
|
||||||
def elem_clicked(self, tag, frac, elem_id, loc):
|
def elem_clicked(self, tag, frac, elem_id, loc):
|
||||||
self.current_frag = elem_id or loc
|
self.current_frag = elem_id or loc
|
||||||
base = _('Location: A <%s> tag inside the file')%tag
|
base = _('Location: A <%s> tag inside the file')%tag
|
||||||
|
@ -14,7 +14,7 @@ from functools import partial
|
|||||||
from PyQt4.Qt import (QPushButton, QFrame, QVariant, QMenu, QInputDialog,
|
from PyQt4.Qt import (QPushButton, QFrame, QVariant, QMenu, QInputDialog,
|
||||||
QDialog, QVBoxLayout, QDialogButtonBox, QSize, QStackedWidget, QWidget,
|
QDialog, QVBoxLayout, QDialogButtonBox, QSize, QStackedWidget, QWidget,
|
||||||
QLabel, Qt, pyqtSignal, QIcon, QTreeWidget, QGridLayout, QTreeWidgetItem,
|
QLabel, Qt, pyqtSignal, QIcon, QTreeWidget, QGridLayout, QTreeWidgetItem,
|
||||||
QToolButton, QItemSelectionModel, QCursor)
|
QToolButton, QItemSelectionModel, QCursor, QKeySequence)
|
||||||
|
|
||||||
from calibre.ebooks.oeb.polish.container import get_container, AZW3Container
|
from calibre.ebooks.oeb.polish.container import get_container, AZW3Container
|
||||||
from calibre.ebooks.oeb.polish.toc import (
|
from calibre.ebooks.oeb.polish.toc import (
|
||||||
@ -27,7 +27,7 @@ from calibre.utils.logging import GUILog
|
|||||||
|
|
||||||
ICON_SIZE = 24
|
ICON_SIZE = 24
|
||||||
|
|
||||||
class XPathDialog(QDialog): # {{{
|
class XPathDialog(QDialog): # {{{
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
QDialog.__init__(self, parent)
|
QDialog.__init__(self, parent)
|
||||||
@ -118,7 +118,7 @@ class XPathDialog(QDialog): # {{{
|
|||||||
return [w.xpath for w in self.widgets if w.xpath.strip()]
|
return [w.xpath for w in self.widgets if w.xpath.strip()]
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class ItemView(QFrame): # {{{
|
class ItemView(QFrame): # {{{
|
||||||
|
|
||||||
add_new_item = pyqtSignal(object, object)
|
add_new_item = pyqtSignal(object, object)
|
||||||
delete_item = pyqtSignal()
|
delete_item = pyqtSignal()
|
||||||
@ -207,7 +207,6 @@ class ItemView(QFrame): # {{{
|
|||||||
)))
|
)))
|
||||||
l.addWidget(b)
|
l.addWidget(b)
|
||||||
|
|
||||||
|
|
||||||
l.addStretch()
|
l.addStretch()
|
||||||
self.w1 = la = QLabel(_('<b>WARNING:</b> calibre only supports the '
|
self.w1 = la = QLabel(_('<b>WARNING:</b> calibre only supports the '
|
||||||
'creation of linear ToCs in AZW3 files. In a '
|
'creation of linear ToCs in AZW3 files. In a '
|
||||||
@ -349,7 +348,9 @@ class ItemView(QFrame): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class TreeWidget(QTreeWidget): # {{{
|
class TreeWidget(QTreeWidget): # {{{
|
||||||
|
|
||||||
|
edit_item = pyqtSignal()
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
QTreeWidget.__init__(self, parent)
|
QTreeWidget.__init__(self, parent)
|
||||||
@ -510,25 +511,30 @@ class TreeWidget(QTreeWidget): # {{{
|
|||||||
|
|
||||||
def show_context_menu(self, point):
|
def show_context_menu(self, point):
|
||||||
item = self.currentItem()
|
item = self.currentItem()
|
||||||
|
def key(k):
|
||||||
|
sc = unicode(QKeySequence(k | Qt.CTRL).toString(QKeySequence.NativeText))
|
||||||
|
return ' [%s]'%sc
|
||||||
|
|
||||||
if item is not None:
|
if item is not None:
|
||||||
m = QMenu()
|
m = QMenu()
|
||||||
ci = unicode(item.data(0, Qt.DisplayRole).toString())
|
ci = unicode(item.data(0, Qt.DisplayRole).toString())
|
||||||
p = item.parent() or self.invisibleRootItem()
|
p = item.parent() or self.invisibleRootItem()
|
||||||
idx = p.indexOfChild(item)
|
idx = p.indexOfChild(item)
|
||||||
if idx > 0:
|
if idx > 0:
|
||||||
m.addAction(QIcon(I('arrow-up.png')), _('Move "%s" up')%ci, self.move_up)
|
m.addAction(QIcon(I('arrow-up.png')), (_('Move "%s" up')%ci)+key(Qt.Key_Up), self.move_up)
|
||||||
if idx + 1 < p.childCount():
|
if idx + 1 < p.childCount():
|
||||||
m.addAction(QIcon(I('arrow-down.png')), _('Move "%s" down')%ci, self.move_down)
|
m.addAction(QIcon(I('arrow-down.png')), (_('Move "%s" down')%ci)+key(Qt.Key_Down), self.move_down)
|
||||||
m.addAction(QIcon(I('trash.png')), _('Remove all selected items'), self.del_items)
|
m.addAction(QIcon(I('trash.png')), _('Remove all selected items'), self.del_items)
|
||||||
if item.parent() is not None:
|
if item.parent() is not None:
|
||||||
m.addAction(QIcon(I('back.png')), _('Unindent "%s"')%ci, self.move_left)
|
m.addAction(QIcon(I('back.png')), (_('Unindent "%s"')%ci)+key(Qt.Key_Left), self.move_left)
|
||||||
if idx > 0:
|
if idx > 0:
|
||||||
m.addAction(QIcon(I('forward.png')), _('Indent "%s"')%ci, self.move_right)
|
m.addAction(QIcon(I('forward.png')), (_('Indent "%s"')%ci)+key(Qt.Key_Right), self.move_right)
|
||||||
|
m.addAction(QIcon(I('edit_input.png')), _('Change the location this entry points to'), self.edit_item)
|
||||||
m.addAction(_('Change all selected items to title case'), self.title_case)
|
m.addAction(_('Change all selected items to title case'), self.title_case)
|
||||||
m.exec_(QCursor.pos())
|
m.exec_(QCursor.pos())
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class TOCView(QWidget): # {{{
|
class TOCView(QWidget): # {{{
|
||||||
|
|
||||||
add_new_item = pyqtSignal(object, object)
|
add_new_item = pyqtSignal(object, object)
|
||||||
|
|
||||||
@ -537,6 +543,7 @@ class TOCView(QWidget): # {{{
|
|||||||
l = self.l = QGridLayout()
|
l = self.l = QGridLayout()
|
||||||
self.setLayout(l)
|
self.setLayout(l)
|
||||||
self.tocw = t = TreeWidget(self)
|
self.tocw = t = TreeWidget(self)
|
||||||
|
self.tocw.edit_item.connect(self.edit_item)
|
||||||
l.addWidget(t, 0, 0, 7, 3)
|
l.addWidget(t, 0, 0, 7, 3)
|
||||||
self.up_button = b = QToolButton(self)
|
self.up_button = b = QToolButton(self)
|
||||||
b.setIcon(QIcon(I('arrow-up.png')))
|
b.setIcon(QIcon(I('arrow-up.png')))
|
||||||
@ -595,6 +602,9 @@ class TOCView(QWidget): # {{{
|
|||||||
|
|
||||||
l.setColumnStretch(2, 10)
|
l.setColumnStretch(2, 10)
|
||||||
|
|
||||||
|
def edit_item(self):
|
||||||
|
self.item_view.edit_item()
|
||||||
|
|
||||||
def event(self, e):
|
def event(self, e):
|
||||||
if e.type() == e.StatusTip:
|
if e.type() == e.StatusTip:
|
||||||
txt = unicode(e.tip()) or self.default_msg
|
txt = unicode(e.tip()) or self.default_msg
|
||||||
@ -742,12 +752,12 @@ class TOCView(QWidget): # {{{
|
|||||||
else:
|
else:
|
||||||
parent = item.parent() or self.root
|
parent = item.parent() or self.root
|
||||||
idx = parent.indexOfChild(item)
|
idx = parent.indexOfChild(item)
|
||||||
if where == 'after': idx += 1
|
if where == 'after':
|
||||||
|
idx += 1
|
||||||
c = self.create_item(parent, child, idx=idx)
|
c = self.create_item(parent, child, idx=idx)
|
||||||
self.tocw.setCurrentItem(c, 0, QItemSelectionModel.ClearAndSelect)
|
self.tocw.setCurrentItem(c, 0, QItemSelectionModel.ClearAndSelect)
|
||||||
self.tocw.scrollToItem(c)
|
self.tocw.scrollToItem(c)
|
||||||
|
|
||||||
|
|
||||||
def create_toc(self):
|
def create_toc(self):
|
||||||
root = TOC()
|
root = TOC()
|
||||||
|
|
||||||
@ -799,7 +809,7 @@ class TOCView(QWidget): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class TOCEditor(QDialog): # {{{
|
class TOCEditor(QDialog): # {{{
|
||||||
|
|
||||||
explode_done = pyqtSignal(object)
|
explode_done = pyqtSignal(object)
|
||||||
writing_done = pyqtSignal(object)
|
writing_done = pyqtSignal(object)
|
||||||
@ -857,6 +867,7 @@ class TOCEditor(QDialog): # {{{
|
|||||||
def accept(self):
|
def accept(self):
|
||||||
if self.stacks.currentIndex() == 2:
|
if self.stacks.currentIndex() == 2:
|
||||||
self.toc_view.update_item(*self.item_edit.result)
|
self.toc_view.update_item(*self.item_edit.result)
|
||||||
|
gprefs['toc_edit_splitter_state'] = bytearray(self.item_edit.splitter.saveState())
|
||||||
self.stacks.setCurrentIndex(1)
|
self.stacks.setCurrentIndex(1)
|
||||||
elif self.stacks.currentIndex() == 1:
|
elif self.stacks.currentIndex() == 1:
|
||||||
self.working = False
|
self.working = False
|
||||||
@ -883,6 +894,7 @@ class TOCEditor(QDialog): # {{{
|
|||||||
if not self.bb.isEnabled():
|
if not self.bb.isEnabled():
|
||||||
return
|
return
|
||||||
if self.stacks.currentIndex() == 2:
|
if self.stacks.currentIndex() == 2:
|
||||||
|
gprefs['toc_edit_splitter_state'] = bytearray(self.item_edit.splitter.saveState())
|
||||||
self.stacks.setCurrentIndex(1)
|
self.stacks.setCurrentIndex(1)
|
||||||
else:
|
else:
|
||||||
self.working = False
|
self.working = False
|
||||||
@ -938,5 +950,5 @@ if __name__ == '__main__':
|
|||||||
d = TOCEditor(sys.argv[-1])
|
d = TOCEditor(sys.argv[-1])
|
||||||
d.start()
|
d.start()
|
||||||
d.exec_()
|
d.exec_()
|
||||||
del d # Needed to prevent sigsegv in exit cleanup
|
del d # Needed to prevent sigsegv in exit cleanup
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ from threading import Thread
|
|||||||
from collections import OrderedDict
|
from collections import OrderedDict
|
||||||
|
|
||||||
from PyQt4.Qt import (Qt, SIGNAL, QTimer, QHelpEvent, QAction,
|
from PyQt4.Qt import (Qt, SIGNAL, QTimer, QHelpEvent, QAction,
|
||||||
QMenu, QIcon, pyqtSignal, QUrl,
|
QMenu, QIcon, pyqtSignal, QUrl, QFont,
|
||||||
QDialog, QSystemTrayIcon, QApplication)
|
QDialog, QSystemTrayIcon, QApplication)
|
||||||
|
|
||||||
from calibre import prints, force_unicode
|
from calibre import prints, force_unicode
|
||||||
@ -47,7 +47,7 @@ from calibre.gui2.proceed import ProceedQuestion
|
|||||||
from calibre.gui2.dialogs.message_box import JobError
|
from calibre.gui2.dialogs.message_box import JobError
|
||||||
from calibre.gui2.job_indicator import Pointer
|
from calibre.gui2.job_indicator import Pointer
|
||||||
|
|
||||||
class Listener(Thread): # {{{
|
class Listener(Thread): # {{{
|
||||||
|
|
||||||
def __init__(self, listener):
|
def __init__(self, listener):
|
||||||
Thread.__init__(self)
|
Thread.__init__(self)
|
||||||
@ -76,7 +76,7 @@ class Listener(Thread): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
class SystemTrayIcon(QSystemTrayIcon): # {{{
|
class SystemTrayIcon(QSystemTrayIcon): # {{{
|
||||||
|
|
||||||
tooltip_requested = pyqtSignal(object)
|
tooltip_requested = pyqtSignal(object)
|
||||||
|
|
||||||
@ -98,7 +98,7 @@ _gui = None
|
|||||||
def get_gui():
|
def get_gui():
|
||||||
return _gui
|
return _gui
|
||||||
|
|
||||||
class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
||||||
TagBrowserMixin, CoverFlowMixin, LibraryViewMixin, SearchBoxMixin,
|
TagBrowserMixin, CoverFlowMixin, LibraryViewMixin, SearchBoxMixin,
|
||||||
SavedSearchBoxMixin, SearchRestrictionMixin, LayoutMixin, UpdateMixin,
|
SavedSearchBoxMixin, SearchRestrictionMixin, LayoutMixin, UpdateMixin,
|
||||||
EbookDownloadMixin
|
EbookDownloadMixin
|
||||||
@ -187,7 +187,6 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
else:
|
else:
|
||||||
stmap[st.name] = st
|
stmap[st.name] = st
|
||||||
|
|
||||||
|
|
||||||
def initialize(self, library_path, db, listener, actions, show_gui=True):
|
def initialize(self, library_path, db, listener, actions, show_gui=True):
|
||||||
opts = self.opts
|
opts = self.opts
|
||||||
self.preferences_action, self.quit_action = actions
|
self.preferences_action, self.quit_action = actions
|
||||||
@ -279,6 +278,7 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
UpdateMixin.__init__(self, opts)
|
UpdateMixin.__init__(self, opts)
|
||||||
|
|
||||||
####################### Search boxes ########################
|
####################### Search boxes ########################
|
||||||
|
SearchRestrictionMixin.__init__(self)
|
||||||
SavedSearchBoxMixin.__init__(self)
|
SavedSearchBoxMixin.__init__(self)
|
||||||
SearchBoxMixin.__init__(self)
|
SearchBoxMixin.__init__(self)
|
||||||
|
|
||||||
@ -313,9 +313,8 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
TagBrowserMixin.__init__(self, db)
|
TagBrowserMixin.__init__(self, db)
|
||||||
|
|
||||||
######################### Search Restriction ##########################
|
######################### Search Restriction ##########################
|
||||||
SearchRestrictionMixin.__init__(self)
|
if db.prefs['virtual_lib_on_startup']:
|
||||||
if db.prefs['gui_restriction']:
|
self.apply_virtual_library(db.prefs['virtual_lib_on_startup'])
|
||||||
self.apply_named_search_restriction(db.prefs['gui_restriction'])
|
|
||||||
|
|
||||||
########################### Cover Flow ################################
|
########################### Cover Flow ################################
|
||||||
|
|
||||||
@ -339,7 +338,6 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
if config['autolaunch_server']:
|
if config['autolaunch_server']:
|
||||||
self.start_content_server()
|
self.start_content_server()
|
||||||
|
|
||||||
|
|
||||||
self.keyboard_interrupt.connect(self.quit, type=Qt.QueuedConnection)
|
self.keyboard_interrupt.connect(self.quit, type=Qt.QueuedConnection)
|
||||||
|
|
||||||
self.read_settings()
|
self.read_settings()
|
||||||
@ -393,7 +391,7 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
if not self.device_manager.is_running('Wireless Devices'):
|
if not self.device_manager.is_running('Wireless Devices'):
|
||||||
error_dialog(self, _('Problem starting the wireless device'),
|
error_dialog(self, _('Problem starting the wireless device'),
|
||||||
_('The wireless device driver did not start. '
|
_('The wireless device driver did not start. '
|
||||||
'It said "%s"')%message, show=True)
|
'It said "%s"')%message, show=True)
|
||||||
self.iactions['Connect Share'].set_smartdevice_action_state()
|
self.iactions['Connect Share'].set_smartdevice_action_state()
|
||||||
|
|
||||||
def start_content_server(self, check_started=True):
|
def start_content_server(self, check_started=True):
|
||||||
@ -494,7 +492,7 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
path = os.path.abspath(argv[1])
|
path = os.path.abspath(argv[1])
|
||||||
if os.access(path, os.R_OK):
|
if os.access(path, os.R_OK):
|
||||||
self.iactions['Add Books'].add_filesystem_book(path)
|
self.iactions['Add Books'].add_filesystem_book(path)
|
||||||
self.setWindowState(self.windowState() & \
|
self.setWindowState(self.windowState() &
|
||||||
~Qt.WindowMinimized|Qt.WindowActive)
|
~Qt.WindowMinimized|Qt.WindowActive)
|
||||||
self.show_windows()
|
self.show_windows()
|
||||||
self.raise_()
|
self.raise_()
|
||||||
@ -526,7 +524,8 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
|
|
||||||
def library_moved(self, newloc, copy_structure=False, call_close=True,
|
def library_moved(self, newloc, copy_structure=False, call_close=True,
|
||||||
allow_rebuild=False):
|
allow_rebuild=False):
|
||||||
if newloc is None: return
|
if newloc is None:
|
||||||
|
return
|
||||||
default_prefs = None
|
default_prefs = None
|
||||||
try:
|
try:
|
||||||
olddb = self.library_view.model().db
|
olddb = self.library_view.model().db
|
||||||
@ -537,7 +536,8 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
try:
|
try:
|
||||||
db = LibraryDatabase2(newloc, default_prefs=default_prefs)
|
db = LibraryDatabase2(newloc, default_prefs=default_prefs)
|
||||||
except (DatabaseException, sqlite.Error):
|
except (DatabaseException, sqlite.Error):
|
||||||
if not allow_rebuild: raise
|
if not allow_rebuild:
|
||||||
|
raise
|
||||||
import traceback
|
import traceback
|
||||||
repair = question_dialog(self, _('Corrupted database'),
|
repair = question_dialog(self, _('Corrupted database'),
|
||||||
_('The library database at %s appears to be corrupted. Do '
|
_('The library database at %s appears to be corrupted. Do '
|
||||||
@ -571,8 +571,8 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
db = self.library_view.model().db
|
db = self.library_view.model().db
|
||||||
self.iactions['Choose Library'].count_changed(db.count())
|
self.iactions['Choose Library'].count_changed(db.count())
|
||||||
self.set_window_title()
|
self.set_window_title()
|
||||||
self.apply_named_search_restriction('') # reset restriction to null
|
self.apply_named_search_restriction('') # reset restriction to null
|
||||||
self.saved_searches_changed(recount=False) # reload the search restrictions combo box
|
self.saved_searches_changed(recount=False) # reload the search restrictions combo box
|
||||||
self.apply_named_search_restriction(db.prefs['gui_restriction'])
|
self.apply_named_search_restriction(db.prefs['gui_restriction'])
|
||||||
for action in self.iactions.values():
|
for action in self.iactions.values():
|
||||||
action.library_changed(db)
|
action.library_changed(db)
|
||||||
@ -596,9 +596,19 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
# interface later
|
# interface later
|
||||||
gc.collect()
|
gc.collect()
|
||||||
|
|
||||||
|
|
||||||
def set_window_title(self):
|
def set_window_title(self):
|
||||||
self.setWindowTitle(__appname__ + u' - || %s ||'%self.iactions['Choose Library'].library_name())
|
db = self.current_db
|
||||||
|
restrictions = [x for x in (db.data.get_base_restriction_name(),
|
||||||
|
db.data.get_search_restriction_name()) if x]
|
||||||
|
restrictions = ' :: '.join(restrictions)
|
||||||
|
font = QFont()
|
||||||
|
if restrictions:
|
||||||
|
restrictions = ' :: ' + restrictions
|
||||||
|
font.setBold(True)
|
||||||
|
self.virtual_library.setFont(font)
|
||||||
|
title = u'{0} - || {1}{2} ||'.format(
|
||||||
|
__appname__, self.iactions['Choose Library'].library_name(), restrictions)
|
||||||
|
self.setWindowTitle(title)
|
||||||
|
|
||||||
def location_selected(self, location):
|
def location_selected(self, location):
|
||||||
'''
|
'''
|
||||||
@ -613,17 +623,15 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
for action in self.iactions.values():
|
for action in self.iactions.values():
|
||||||
action.location_selected(location)
|
action.location_selected(location)
|
||||||
if location == 'library':
|
if location == 'library':
|
||||||
self.search_restriction.setEnabled(True)
|
self.virtual_library_menu.setEnabled(True)
|
||||||
self.highlight_only_button.setEnabled(True)
|
self.highlight_only_button.setEnabled(True)
|
||||||
else:
|
else:
|
||||||
self.search_restriction.setEnabled(False)
|
self.virtual_library_menu.setEnabled(False)
|
||||||
self.highlight_only_button.setEnabled(False)
|
self.highlight_only_button.setEnabled(False)
|
||||||
# Reset the view in case something changed while it was invisible
|
# Reset the view in case something changed while it was invisible
|
||||||
self.current_view().reset()
|
self.current_view().reset()
|
||||||
self.set_number_of_books_shown()
|
self.set_number_of_books_shown()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def job_exception(self, job, dialog_title=_('Conversion Error')):
|
def job_exception(self, job, dialog_title=_('Conversion Error')):
|
||||||
if not hasattr(self, '_modeless_dialogs'):
|
if not hasattr(self, '_modeless_dialogs'):
|
||||||
self._modeless_dialogs = []
|
self._modeless_dialogs = []
|
||||||
@ -715,7 +723,7 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
self.read_layout_settings()
|
self.read_layout_settings()
|
||||||
|
|
||||||
def write_settings(self):
|
def write_settings(self):
|
||||||
with gprefs: # Only write to gprefs once
|
with gprefs: # Only write to gprefs once
|
||||||
config.set('main_window_geometry', self.saveGeometry())
|
config.set('main_window_geometry', self.saveGeometry())
|
||||||
dynamic.set('sort_history', self.library_view.model().sort_history)
|
dynamic.set('sort_history', self.library_view.model().sort_history)
|
||||||
self.save_layout_state()
|
self.save_layout_state()
|
||||||
@ -748,7 +756,6 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
def shutdown(self, write_settings=True):
|
def shutdown(self, write_settings=True):
|
||||||
try:
|
try:
|
||||||
db = self.library_view.model().db
|
db = self.library_view.model().db
|
||||||
@ -808,13 +815,11 @@ class Main(MainWindow, MainWindowMixin, DeviceMixin, EmailMixin, # {{{
|
|||||||
pass
|
pass
|
||||||
QApplication.instance().quit()
|
QApplication.instance().quit()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def closeEvent(self, e):
|
def closeEvent(self, e):
|
||||||
self.write_settings()
|
self.write_settings()
|
||||||
if self.system_tray_icon.isVisible():
|
if self.system_tray_icon.isVisible():
|
||||||
if not dynamic['systray_msg'] and not isosx:
|
if not dynamic['systray_msg'] and not isosx:
|
||||||
info_dialog(self, 'calibre', 'calibre '+ \
|
info_dialog(self, 'calibre', 'calibre '+
|
||||||
_('will keep running in the system tray. To close it, '
|
_('will keep running in the system tray. To close it, '
|
||||||
'choose <b>Quit</b> in the context menu of the '
|
'choose <b>Quit</b> in the context menu of the '
|
||||||
'system tray.'), show_copy_button=False).exec_()
|
'system tray.'), show_copy_button=False).exec_()
|
||||||
|
@ -209,7 +209,8 @@ class ResultCache(SearchQueryParser): # {{{
|
|||||||
self._data = []
|
self._data = []
|
||||||
self._map = self._map_filtered = []
|
self._map = self._map_filtered = []
|
||||||
self.first_sort = True
|
self.first_sort = True
|
||||||
self.search_restriction = ''
|
self.search_restriction = self.base_restriction = ''
|
||||||
|
self.base_restriction_name = self.search_restriction_name = ''
|
||||||
self.search_restriction_book_count = 0
|
self.search_restriction_book_count = 0
|
||||||
self.marked_ids_dict = {}
|
self.marked_ids_dict = {}
|
||||||
self.field_metadata = field_metadata
|
self.field_metadata = field_metadata
|
||||||
@ -825,8 +826,19 @@ class ResultCache(SearchQueryParser): # {{{
|
|||||||
return ans
|
return ans
|
||||||
self._map_filtered = ans
|
self._map_filtered = ans
|
||||||
|
|
||||||
|
def _build_restriction_string(self, restriction):
|
||||||
|
if self.base_restriction:
|
||||||
|
if restriction:
|
||||||
|
return u'(%s) and (%s)' % (self.base_restriction, restriction)
|
||||||
|
else:
|
||||||
|
return self.base_restriction
|
||||||
|
else:
|
||||||
|
return restriction
|
||||||
|
|
||||||
def search_getting_ids(self, query, search_restriction,
|
def search_getting_ids(self, query, search_restriction,
|
||||||
set_restriction_count=False):
|
set_restriction_count=False, use_virtual_library=True):
|
||||||
|
if use_virtual_library:
|
||||||
|
search_restriction = self._build_restriction_string(search_restriction)
|
||||||
q = ''
|
q = ''
|
||||||
if not query or not query.strip():
|
if not query or not query.strip():
|
||||||
q = search_restriction
|
q = search_restriction
|
||||||
@ -847,11 +859,32 @@ class ResultCache(SearchQueryParser): # {{{
|
|||||||
self.search_restriction_book_count = len(rv)
|
self.search_restriction_book_count = len(rv)
|
||||||
return rv
|
return rv
|
||||||
|
|
||||||
|
def get_search_restriction(self):
|
||||||
|
return self.search_restriction
|
||||||
|
|
||||||
def set_search_restriction(self, s):
|
def set_search_restriction(self, s):
|
||||||
self.search_restriction = s
|
self.search_restriction = s
|
||||||
|
|
||||||
|
def get_base_restriction(self):
|
||||||
|
return self.base_restriction
|
||||||
|
|
||||||
|
def set_base_restriction(self, s):
|
||||||
|
self.base_restriction = s
|
||||||
|
|
||||||
|
def get_base_restriction_name(self):
|
||||||
|
return self.base_restriction_name
|
||||||
|
|
||||||
|
def set_base_restriction_name(self, s):
|
||||||
|
self.base_restriction_name = s
|
||||||
|
|
||||||
|
def get_search_restriction_name(self):
|
||||||
|
return self.search_restriction_name
|
||||||
|
|
||||||
|
def set_search_restriction_name(self, s):
|
||||||
|
self.search_restriction_name = s
|
||||||
|
|
||||||
def search_restriction_applied(self):
|
def search_restriction_applied(self):
|
||||||
return bool(self.search_restriction)
|
return bool(self.search_restriction) or bool((self.base_restriction))
|
||||||
|
|
||||||
def get_search_restriction_book_count(self):
|
def get_search_restriction_book_count(self):
|
||||||
return self.search_restriction_book_count
|
return self.search_restriction_book_count
|
||||||
@ -1002,7 +1035,7 @@ class ResultCache(SearchQueryParser): # {{{
|
|||||||
if field is not None:
|
if field is not None:
|
||||||
self.sort(field, ascending)
|
self.sort(field, ascending)
|
||||||
self._map_filtered = list(self._map)
|
self._map_filtered = list(self._map)
|
||||||
if self.search_restriction:
|
if self.search_restriction or self.base_restriction:
|
||||||
self.search('', return_matches=False)
|
self.search('', return_matches=False)
|
||||||
|
|
||||||
# Sorting functions {{{
|
# Sorting functions {{{
|
||||||
|
@ -229,6 +229,8 @@ class LibraryDatabase2(LibraryDatabase, SchemaUpgrade, CustomColumns):
|
|||||||
('uuid', False), ('comments', True), ('id', False), ('pubdate', False),
|
('uuid', False), ('comments', True), ('id', False), ('pubdate', False),
|
||||||
('last_modified', False), ('size', False), ('languages', False),
|
('last_modified', False), ('size', False), ('languages', False),
|
||||||
]
|
]
|
||||||
|
defs['virtual_libraries'] = {}
|
||||||
|
defs['virtual_lib_on_startup'] = defs['cs_virtual_lib_on_startup'] = ''
|
||||||
|
|
||||||
# Migrate the bool tristate tweak
|
# Migrate the bool tristate tweak
|
||||||
defs['bools_are_tristate'] = \
|
defs['bools_are_tristate'] = \
|
||||||
@ -279,6 +281,24 @@ class LibraryDatabase2(LibraryDatabase, SchemaUpgrade, CustomColumns):
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
# migrate the gui_restriction preference to a virtual library
|
||||||
|
gr_pref = self.prefs.get('gui_restriction', None)
|
||||||
|
if gr_pref:
|
||||||
|
virt_libs = self.prefs.get('virtual_libraries', {})
|
||||||
|
virt_libs[gr_pref] = 'search:"' + gr_pref + '"'
|
||||||
|
self.prefs['virtual_libraries'] = virt_libs
|
||||||
|
self.prefs['gui_restriction'] = ''
|
||||||
|
self.prefs['virtual_lib_on_startup'] = gr_pref
|
||||||
|
|
||||||
|
# migrate the cs_restriction preference to a virtual library
|
||||||
|
gr_pref = self.prefs.get('cs_restriction', None)
|
||||||
|
if gr_pref:
|
||||||
|
virt_libs = self.prefs.get('virtual_libraries', {})
|
||||||
|
virt_libs[gr_pref] = 'search:"' + gr_pref + '"'
|
||||||
|
self.prefs['virtual_libraries'] = virt_libs
|
||||||
|
self.prefs['cs_restriction'] = ''
|
||||||
|
self.prefs['cs_virtual_lib_on_startup'] = gr_pref
|
||||||
|
|
||||||
# Rename any user categories with names that differ only in case
|
# Rename any user categories with names that differ only in case
|
||||||
user_cats = self.prefs.get('user_categories', [])
|
user_cats = self.prefs.get('user_categories', [])
|
||||||
catmap = {}
|
catmap = {}
|
||||||
|
@ -205,26 +205,32 @@ class LibraryServer(ContentServer, MobileServer, XMLServer, OPDSServer, Cache,
|
|||||||
|
|
||||||
def set_database(self, db):
|
def set_database(self, db):
|
||||||
self.db = db
|
self.db = db
|
||||||
|
virt_libs = db.prefs.get('virtual_libraries', {})
|
||||||
sr = getattr(self.opts, 'restriction', None)
|
sr = getattr(self.opts, 'restriction', None)
|
||||||
sr = db.prefs.get('cs_restriction', '') if sr is None else sr
|
if sr:
|
||||||
self.set_search_restriction(sr)
|
if sr in virt_libs:
|
||||||
|
sr = virt_libs[sr]
|
||||||
|
elif sr not in saved_searches().names():
|
||||||
|
prints('WARNING: Content server: search restriction ',
|
||||||
|
sr, ' does not exist')
|
||||||
|
sr = ''
|
||||||
|
else:
|
||||||
|
sr = 'search:"%s"'%sr
|
||||||
|
else:
|
||||||
|
sr = db.prefs.get('cs_virtual_lib_on_startup', '')
|
||||||
|
if sr:
|
||||||
|
if sr not in virt_libs:
|
||||||
|
prints('WARNING: Content server: virtual library ',
|
||||||
|
sr, ' does not exist')
|
||||||
|
sr = ''
|
||||||
|
else:
|
||||||
|
sr = virt_libs[sr]
|
||||||
|
self.search_restriction = sr
|
||||||
|
self.reset_caches()
|
||||||
|
|
||||||
def graceful(self):
|
def graceful(self):
|
||||||
cherrypy.engine.graceful()
|
cherrypy.engine.graceful()
|
||||||
|
|
||||||
def set_search_restriction(self, restriction):
|
|
||||||
self.search_restriction_name = restriction
|
|
||||||
if restriction:
|
|
||||||
if restriction not in saved_searches().names():
|
|
||||||
prints('WARNING: Content server: search restriction ',
|
|
||||||
restriction, ' does not exist')
|
|
||||||
self.search_restriction = ''
|
|
||||||
else:
|
|
||||||
self.search_restriction = 'search:"%s"'%restriction
|
|
||||||
else:
|
|
||||||
self.search_restriction = ''
|
|
||||||
self.reset_caches()
|
|
||||||
|
|
||||||
def setup_loggers(self):
|
def setup_loggers(self):
|
||||||
access_file = log_access_file
|
access_file = log_access_file
|
||||||
error_file = log_error_file
|
error_file = log_error_file
|
||||||
|
@ -145,10 +145,7 @@ def render_rating(rating, url_prefix, container='span', prefix=None): # {{{
|
|||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
def get_category_items(category, items, restriction, datatype, prefix): # {{{
|
def get_category_items(category, items, datatype, prefix): # {{{
|
||||||
|
|
||||||
if category == 'search':
|
|
||||||
items = [x for x in items if x.name != restriction]
|
|
||||||
|
|
||||||
def item(i):
|
def item(i):
|
||||||
templ = (u'<div title="{4}" class="category-item">'
|
templ = (u'<div title="{4}" class="category-item">'
|
||||||
@ -489,8 +486,7 @@ class BrowseServer(object):
|
|||||||
if not cats and len(items) == 1:
|
if not cats and len(items) == 1:
|
||||||
# Only one item in category, go directly to book list
|
# Only one item in category, go directly to book list
|
||||||
html = get_category_items(category, items,
|
html = get_category_items(category, items,
|
||||||
self.search_restriction_name, datatype,
|
datatype, self.opts.url_prefix)
|
||||||
self.opts.url_prefix)
|
|
||||||
href = re.search(r'<a href="([^"]+)"', html)
|
href = re.search(r'<a href="([^"]+)"', html)
|
||||||
if href is not None:
|
if href is not None:
|
||||||
raise cherrypy.HTTPRedirect(href.group(1))
|
raise cherrypy.HTTPRedirect(href.group(1))
|
||||||
@ -498,8 +494,7 @@ class BrowseServer(object):
|
|||||||
if len(items) <= self.opts.max_opds_ungrouped_items:
|
if len(items) <= self.opts.max_opds_ungrouped_items:
|
||||||
script = 'false'
|
script = 'false'
|
||||||
items = get_category_items(category, items,
|
items = get_category_items(category, items,
|
||||||
self.search_restriction_name, datatype,
|
datatype, self.opts.url_prefix)
|
||||||
self.opts.url_prefix)
|
|
||||||
else:
|
else:
|
||||||
getter = lambda x: unicode(getattr(x, 'sort', x.name))
|
getter = lambda x: unicode(getattr(x, 'sort', x.name))
|
||||||
starts = set([])
|
starts = set([])
|
||||||
@ -588,8 +583,7 @@ class BrowseServer(object):
|
|||||||
|
|
||||||
sort = self.browse_sort_categories(entries, sort)
|
sort = self.browse_sort_categories(entries, sort)
|
||||||
entries = get_category_items(category, entries,
|
entries = get_category_items(category, entries,
|
||||||
self.search_restriction_name, datatype,
|
datatype, self.opts.url_prefix)
|
||||||
self.opts.url_prefix)
|
|
||||||
return json.dumps(entries, ensure_ascii=True)
|
return json.dumps(entries, ensure_ascii=True)
|
||||||
|
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@ def stop_threaded_server(server):
|
|||||||
server.exit()
|
server.exit()
|
||||||
server.thread = None
|
server.thread = None
|
||||||
|
|
||||||
def create_wsgi_app(path_to_library=None, prefix=''):
|
def create_wsgi_app(path_to_library=None, prefix='', virtual_library=None):
|
||||||
'WSGI entry point'
|
'WSGI entry point'
|
||||||
from calibre.library import db
|
from calibre.library import db
|
||||||
cherrypy.config.update({'environment': 'embedded'})
|
cherrypy.config.update({'environment': 'embedded'})
|
||||||
@ -32,6 +32,7 @@ def create_wsgi_app(path_to_library=None, prefix=''):
|
|||||||
parser = option_parser()
|
parser = option_parser()
|
||||||
opts, args = parser.parse_args(['calibre-server'])
|
opts, args = parser.parse_args(['calibre-server'])
|
||||||
opts.url_prefix = prefix
|
opts.url_prefix = prefix
|
||||||
|
opts.restriction = virtual_library
|
||||||
server = LibraryServer(db, opts, wsgi=True, show_tracebacks=True)
|
server = LibraryServer(db, opts, wsgi=True, show_tracebacks=True)
|
||||||
return cherrypy.Application(server, script_name=None, config=server.config)
|
return cherrypy.Application(server, script_name=None, config=server.config)
|
||||||
|
|
||||||
@ -54,10 +55,11 @@ The OPDS interface is advertised via BonJour automatically.
|
|||||||
help=_('Write process PID to the specified file'))
|
help=_('Write process PID to the specified file'))
|
||||||
parser.add_option('--daemonize', default=False, action='store_true',
|
parser.add_option('--daemonize', default=False, action='store_true',
|
||||||
help='Run process in background as a daemon. No effect on windows.')
|
help='Run process in background as a daemon. No effect on windows.')
|
||||||
parser.add_option('--restriction', default=None,
|
parser.add_option('--restriction', '--virtual-library', default=None,
|
||||||
help=_('Specifies a restriction to be used for this invocation. '
|
help=_('Specifies a virtual library to be used for this invocation. '
|
||||||
'This option overrides any per-library settings specified'
|
'This option overrides any per-library settings specified'
|
||||||
' in the GUI'))
|
' in the GUI. For compatibility, if the value is not a '
|
||||||
|
'virtual library but is a saved search, that saved search is used.'))
|
||||||
parser.add_option('--auto-reload', default=False, action='store_true',
|
parser.add_option('--auto-reload', default=False, action='store_true',
|
||||||
help=_('Auto reload server when source code changes. May not'
|
help=_('Auto reload server when source code changes. May not'
|
||||||
' work in all environments.'))
|
' work in all environments.'))
|
||||||
@ -97,7 +99,6 @@ def daemonize(stdin='/dev/null', stdout='/dev/null', stderr='/dev/null'):
|
|||||||
os.dup2(se.fileno(), sys.stderr.fileno())
|
os.dup2(se.fileno(), sys.stderr.fileno())
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def main(args=sys.argv):
|
def main(args=sys.argv):
|
||||||
from calibre.library.database2 import LibraryDatabase2
|
from calibre.library.database2 import LibraryDatabase2
|
||||||
parser = option_parser()
|
parser = option_parser()
|
||||||
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user