Compare commits

..

11 Commits

Author SHA1 Message Date
Alexei Yatskov
2d710ee74e
Merge pull request #42 from axu2/options
Add new fit, fill, stretch options
2022-07-16 09:38:54 -07:00
Alexei Yatskov
5a5188ee52
Merge pull request #51 from axu2/rotate90
use proper rot90
2022-07-14 14:42:11 -07:00
2c8a539f39 Update README 2022-07-14 14:17:34 -07:00
Alex Xu
70656272d1 Merge branch 'master' into options 2022-04-23 19:30:26 -07:00
Alex Xu
383cf86076 use proper rot90 2022-02-19 14:57:36 -08:00
Alex Yatskov
0dbc45fc41
Merge pull request #50 from axu2/patch-2
don't rotate small images
2022-02-17 19:28:53 -08:00
Alex Xu
cebfde3174
don't rotate small images 2022-02-13 16:24:40 -08:00
Alex Yatskov
aeb813f76b
Merge pull request #49 from axu2/crop
Don't crop 2 page spreads
2022-02-05 20:39:53 -08:00
Alex Xu
7df96e86eb
Make ratio comparison more explicit 2022-01-17 13:47:47 -08:00
Alex Xu
b70da17f80 remove extra change 2022-01-07 19:26:01 -08:00
Alex Xu
08a3f4f075 Add new options 2021-04-24 21:53:48 -07:00
5 changed files with 72 additions and 22 deletions

3
.gitignore vendored
View File

@ -145,6 +145,9 @@ cython_debug/
*.cbz
*.pdf
# vscode
.vscode/
*.bat
.idea/

View File

@ -1,12 +1,3 @@
<!-- +++
Area = "projects"
GitHub = "mangle"
Layout = "page"
Tags = ["kindle", "manga", "mangle", "pil", "pyqt", "python", "gpl license"]
Description = "Manga processor for the Kindle e-book reader."
Collection = "ProjectsComplete"
+++ -->
# Mangle
Mangle is a cross-platform image converter and optimizer built for reading Manga on the Amazon Kindle and other E-ink

View File

@ -24,12 +24,14 @@ class ImageFlags:
Resize = 1 << 1
Frame = 1 << 2
Quantize = 1 << 3
ScaleCrop = 1 << 4
Fit = 1 << 4
SplitRightLeft = 1 << 5 # split right then left
SplitRight = 1 << 6 # split only the right page
SplitLeft = 1 << 7 # split only the left page
SplitLeftRight = 1 << 8 # split left then right page
AutoCrop = 1 << 9
Fill = 1 << 10
Stretch = 1 << 11
class KindleData:
@ -134,17 +136,45 @@ def quantizeImage(image, palette):
@protect_bad_image
def scaleCropImage(image, size):
def fitImage(image, size, method=Image.ANTIALIAS):
# copied from ImageOps.contain() from the Python3 version of Pillow
# with division related modifications for Python2
im_ratio = 1.0 * image.width / image.height
dest_ratio = 1.0 * size[0] / size[1]
if im_ratio != dest_ratio:
if im_ratio > dest_ratio:
new_height = int(1.0 * image.height / image.width * size[0])
if new_height != size[1]:
size = (size[0], new_height)
else:
new_width = int(1.0 * image.width / image.height * size[1])
if new_width != size[0]:
size = (new_width, size[1])
return image.resize(size, resample=method)
@protect_bad_image
def fillImage(image, size):
widthDev, heightDev = size
widthImg, heightImg = image.size
imgRatio = float(widthImg) / float(heightImg)
devRatio = float(widthDev) / float(heightDev)
# don't crop 2 page spreads.
if (float(widthImg) / float(heightImg)) > (float(widthDev) / float(heightDev)):
if imgRatio > devRatio:
return resizeImage(image, size)
return ImageOps.fit(image, size, Image.ANTIALIAS)
@protect_bad_image
def stretchImage(image, size):
return image.resize(size, Image.ANTIALIAS)
@protect_bad_image
def resizeImage(image, size):
widthDev, heightDev = size
@ -182,8 +212,11 @@ def orientImage(image, size):
widthDev, heightDev = size
widthImg, heightImg = image.size
if widthImg <= widthDev and heightImg <= heightDev:
return image
if (widthImg > heightImg) != (widthDev > heightDev):
return image.rotate(90, Image.BICUBIC, True)
return image.transpose(Image.ROTATE_90)
return image
@ -191,12 +224,11 @@ def orientImage(image, size):
# by inverting colors, and asking a bounder box ^^
@protect_bad_image
def autoCropImage(image):
w, h = image.size
try:
x0, y0, xend, yend = ImageChops.invert(image).getbbox()
except TypeError: # bad image, specific to chops
return image
image = image.crop((0, y0, w, yend))
image = image.crop((x0, y0, xend, yend))
return image
@ -283,8 +315,12 @@ def convertImage(source, target, device, flags):
image = orientImage(image, size)
if flags & ImageFlags.Resize:
image = resizeImage(image, size)
if flags & ImageFlags.ScaleCrop:
image = scaleCropImage(image, size)
if flags & ImageFlags.Fit:
image = fitImage(image, size)
if flags & ImageFlags.Fill:
image = fillImage(image, size)
if flags & ImageFlags.Stretch:
image = stretchImage(image, size)
if flags & ImageFlags.Frame:
image = frameImage(image, tuple(palette[:3]), tuple(palette[-3:]), size)
if flags & ImageFlags.Quantize:

View File

@ -43,7 +43,9 @@ class DialogOptions(QtGui.QDialog):
self.checkboxOverwrite.setChecked(self.book.overwrite)
self.checkboxOrient.setChecked(self.book.imageFlags & ImageFlags.Orient)
self.checkboxResize.setChecked(self.book.imageFlags & ImageFlags.Resize)
self.checkboxScaleCrop.setChecked(self.book.imageFlags & ImageFlags.ScaleCrop)
self.checkboxFit.setChecked(self.book.imageFlags & ImageFlags.Fit)
self.checkboxFill.setChecked(self.book.imageFlags & ImageFlags.Fill)
self.checkboxStretch.setChecked(self.book.imageFlags & ImageFlags.Stretch)
self.checkboxQuantize.setChecked(self.book.imageFlags & ImageFlags.Quantize)
self.checkboxFrame.setChecked(self.book.imageFlags & ImageFlags.Frame)
@ -62,8 +64,12 @@ class DialogOptions(QtGui.QDialog):
imageFlags |= ImageFlags.Orient
if self.checkboxResize.isChecked():
imageFlags |= ImageFlags.Resize
if self.checkboxScaleCrop.isChecked():
imageFlags |= ImageFlags.ScaleCrop
if self.checkboxFit.isChecked():
imageFlags |= ImageFlags.Fit
if self.checkboxFill.isChecked():
imageFlags |= ImageFlags.Fill
if self.checkboxStretch.isChecked():
imageFlags |= ImageFlags.Stretch
if self.checkboxQuantize.isChecked():
imageFlags |= ImageFlags.Quantize
if self.checkboxFrame.isChecked():

View File

@ -215,9 +215,23 @@
</widget>
</item>
<item>
<widget class="QRadioButton" name="checkboxScaleCrop">
<widget class="QRadioButton" name="checkboxFit">
<property name="text">
<string>Scale and crop images to fill screen width</string>
<string>Fit to screen with borders</string>
</property>
</widget>
</item>
<item>
<widget class="QRadioButton" name="checkboxFill">
<property name="text">
<string>Fill screen by cropping</string>
</property>
</widget>
</item>
<item>
<widget class="QRadioButton" name="checkboxStretch">
<property name="text">
<string>Stretch images to fill screen</string>
</property>
</widget>
</item>