mirror of
https://github.com/ciromattia/kcc
synced 2026-04-17 22:48:53 +00:00
Compare commits
1 Commits
v9.5.0
...
axu2-patch
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3920b5c577 |
@@ -1,40 +1,13 @@
|
|||||||
.git
|
.git
|
||||||
.github
|
.github
|
||||||
|
|
||||||
build
|
build
|
||||||
dist
|
dist
|
||||||
KindleComicConverter.egg-info
|
KindleComicConverter.egg-info
|
||||||
|
|
||||||
.dockerignore
|
.dockerignore
|
||||||
.gitignore
|
.gitignore
|
||||||
.travis.yml
|
.travis.yml
|
||||||
|
|
||||||
Dockerfile
|
Dockerfile
|
||||||
venv
|
venv
|
||||||
.venv
|
|
||||||
__pycache__/
|
|
||||||
*/__pycache__/
|
|
||||||
*.pyc
|
|
||||||
|
|
||||||
*.md
|
*.md
|
||||||
*.txt
|
LICENSE.txt
|
||||||
!requirements-docker.txt
|
|
||||||
MANIFEST.in
|
MANIFEST.in
|
||||||
|
|
||||||
*.yml
|
|
||||||
*.spec
|
|
||||||
|
|
||||||
*.svg
|
|
||||||
*.jpg
|
|
||||||
*.json
|
|
||||||
|
|
||||||
gen_ui_files.bat
|
|
||||||
gen_ui_files.sh
|
|
||||||
|
|
||||||
gui/
|
|
||||||
icons/
|
|
||||||
|
|
||||||
kindlecomicconverter/KCC_gui.py
|
|
||||||
kindlecomicconverter/KCC_rc.py
|
|
||||||
kindlecomicconverter/KCC_ui_editor.py
|
|
||||||
kindlecomicconverter/KCC_ui.py
|
|
||||||
|
|||||||
15
.github/FUNDING.yml
vendored
15
.github/FUNDING.yml
vendored
@@ -1,15 +0,0 @@
|
|||||||
# These are supported funding model platforms
|
|
||||||
|
|
||||||
github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2]
|
|
||||||
patreon: # Replace with a single Patreon username
|
|
||||||
open_collective: # Replace with a single Open Collective username
|
|
||||||
ko_fi: eink_dude
|
|
||||||
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
|
||||||
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
|
||||||
liberapay: # Replace with a single Liberapay username
|
|
||||||
issuehunt: # Replace with a single IssueHunt username
|
|
||||||
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
|
|
||||||
polar: # Replace with a single Polar username
|
|
||||||
buy_me_a_coffee: # Replace with a single Buy Me a Coffee username
|
|
||||||
thanks_dev: # Replace with a single thanks.dev username
|
|
||||||
custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2']
|
|
||||||
8
.github/workflows/codeql-analysis.yml
vendored
8
.github/workflows/codeql-analysis.yml
vendored
@@ -38,11 +38,11 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v6
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v4
|
uses: github/codeql-action/init@v3
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
# If you wish to specify custom queries, you can do so here or in a config file.
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
@@ -56,7 +56,7 @@ jobs:
|
|||||||
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
||||||
# If this step fails, then you should remove it and run the build manually (see below)
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v4
|
uses: github/codeql-action/autobuild@v3
|
||||||
|
|
||||||
# ℹ️ Command-line programs to run using the OS shell.
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
# 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun
|
# 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun
|
||||||
@@ -69,6 +69,6 @@ jobs:
|
|||||||
# ./location_of_script_within_repo/buildscript.sh
|
# ./location_of_script_within_repo/buildscript.sh
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v4
|
uses: github/codeql-action/analyze@v3
|
||||||
with:
|
with:
|
||||||
category: "/language:${{matrix.language}}"
|
category: "/language:${{matrix.language}}"
|
||||||
|
|||||||
34
.github/workflows/docker-base-publish.yml
vendored
Normal file
34
.github/workflows/docker-base-publish.yml
vendored
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
name: Docker base
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
tags: [ 'docker-base-*' ]
|
||||||
|
|
||||||
|
# Don't trigger if it's just a documentation update
|
||||||
|
paths-ignore:
|
||||||
|
- '**.md'
|
||||||
|
- '**.MD'
|
||||||
|
- '**.yml'
|
||||||
|
- 'docs/**'
|
||||||
|
- 'LICENSE'
|
||||||
|
- '.gitattributes'
|
||||||
|
- '.gitignore'
|
||||||
|
- '.dockerignore'
|
||||||
|
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build_and_push:
|
||||||
|
uses: sdr-enthusiasts/common-github-workflows/.github/workflows/build_and_push_image.yml@main
|
||||||
|
with:
|
||||||
|
docker_build_file: ./Dockerfile-base
|
||||||
|
platform_linux_arm32v7_enabled: true
|
||||||
|
platform_linux_arm64v8_enabled: true
|
||||||
|
platform_linux_amd64_enabled: true
|
||||||
|
push_enabled: true
|
||||||
|
build_nohealthcheck: false
|
||||||
|
ghcr_repo_owner: ${{ github.repository_owner }}
|
||||||
|
ghcr_repo: ${{ github.repository }}
|
||||||
|
build_latest: false
|
||||||
|
secrets:
|
||||||
|
ghcr_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
68
.github/workflows/docker-publish.yml
vendored
68
.github/workflows/docker-publish.yml
vendored
@@ -1,10 +1,10 @@
|
|||||||
name: Build and Publish Docker Image
|
name: Docker
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
push:
|
push:
|
||||||
tags:
|
# Publish semver tags as releases.
|
||||||
- 'v*.*.*'
|
tags: [ 'v*.*.*' ]
|
||||||
|
|
||||||
# Don't trigger if it's just a documentation update
|
# Don't trigger if it's just a documentation update
|
||||||
paths-ignore:
|
paths-ignore:
|
||||||
@@ -15,53 +15,19 @@ on:
|
|||||||
- 'LICENSE'
|
- 'LICENSE'
|
||||||
- '.gitattributes'
|
- '.gitattributes'
|
||||||
- '.gitignore'
|
- '.gitignore'
|
||||||
|
- '.dockerignore'
|
||||||
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build_and_publish_base_image:
|
build_and_push:
|
||||||
runs-on: ubuntu-latest
|
uses: sdr-enthusiasts/common-github-workflows/.github/workflows/build_and_push_image.yml@main
|
||||||
steps:
|
with:
|
||||||
- name: Checkout
|
platform_linux_arm32v7_enabled: true
|
||||||
uses: actions/checkout@v6
|
platform_linux_arm64v8_enabled: true
|
||||||
|
platform_linux_amd64_enabled: true
|
||||||
- name: Login to GitHub Container Registry
|
push_enabled: true
|
||||||
uses: docker/login-action@v3
|
build_nohealthcheck: false
|
||||||
with:
|
ghcr_repo_owner: ${{ github.repository_owner }}
|
||||||
registry: ghcr.io
|
ghcr_repo: ${{ github.repository }}
|
||||||
username: ${{ github.repository_owner }}
|
secrets:
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
ghcr_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
- name: Set up QEMU
|
|
||||||
uses: docker/setup-qemu-action@v3
|
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
|
||||||
uses: docker/setup-buildx-action@v3
|
|
||||||
|
|
||||||
- name: Set Release Date
|
|
||||||
id: release_date
|
|
||||||
run: |
|
|
||||||
echo "release_date=$(date --rfc-3339=date)" >> $GITHUB_OUTPUT
|
|
||||||
|
|
||||||
- name: Docker meta
|
|
||||||
id: meta
|
|
||||||
uses: docker/metadata-action@v5
|
|
||||||
with:
|
|
||||||
images: ghcr.io/${{ github.repository_owner }}/kcc
|
|
||||||
# Always creates the "latest" tag
|
|
||||||
flavor: |
|
|
||||||
latest=true
|
|
||||||
tags: |
|
|
||||||
type=ref,event=tag
|
|
||||||
type=raw,value=${{ steps.release_date.outputs.release_date }}
|
|
||||||
|
|
||||||
- name: Build and push
|
|
||||||
uses: docker/build-push-action@v6
|
|
||||||
with:
|
|
||||||
platforms: linux/amd64,linux/arm64,linux/arm/v7
|
|
||||||
context: .
|
|
||||||
push: true
|
|
||||||
tags: |
|
|
||||||
${{ steps.meta.outputs.tags }}
|
|
||||||
cache-from: |
|
|
||||||
type=registry,ref=ghcr.io/ciromattia/kcc:cache
|
|
||||||
type=registry,ref=ghcr.io/${{ github.repository_owner }}/kcc:cache
|
|
||||||
cache-to: type=registry,ref=ghcr.io/${{ github.repository_owner }}/kcc:cache,mode=max
|
|
||||||
|
|||||||
9
.github/workflows/package-linux.yml
vendored
9
.github/workflows/package-linux.yml
vendored
@@ -25,9 +25,9 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v6
|
- uses: actions/checkout@v4
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v6
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: 3.11
|
python-version: 3.11
|
||||||
cache: 'pip'
|
cache: 'pip'
|
||||||
@@ -59,7 +59,7 @@ jobs:
|
|||||||
env:
|
env:
|
||||||
UPDATE_INFO: gh-releases-zsync|ciromattia|kcc|latest|*x86_64.AppImage.zsync
|
UPDATE_INFO: gh-releases-zsync|ciromattia|kcc|latest|*x86_64.AppImage.zsync
|
||||||
- name: upload artifact
|
- name: upload artifact
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: AppImage
|
name: AppImage
|
||||||
path: './*.AppImage*'
|
path: './*.AppImage*'
|
||||||
@@ -68,7 +68,8 @@ jobs:
|
|||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
with:
|
with:
|
||||||
prerelease: true
|
prerelease: true
|
||||||
generate_release_notes: false
|
generate_release_notes: true
|
||||||
files: |
|
files: |
|
||||||
|
CHANGELOG.md
|
||||||
LICENSE.txt
|
LICENSE.txt
|
||||||
*.AppImage*
|
*.AppImage*
|
||||||
|
|||||||
16
.github/workflows/package-macos.yml
vendored
16
.github/workflows/package-macos.yml
vendored
@@ -25,14 +25,12 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [ macos-15-intel, macos-14 ]
|
os: [ macos-13, macos-14 ]
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
env:
|
|
||||||
MACOSX_DEPLOYMENT_TARGET: '14.0'
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v6
|
- uses: actions/checkout@v4
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v6
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: 3.11
|
python-version: 3.11
|
||||||
cache: 'pip'
|
cache: 'pip'
|
||||||
@@ -71,7 +69,7 @@ jobs:
|
|||||||
# apply provisioning profile
|
# apply provisioning profile
|
||||||
mkdir -p ~/Library/MobileDevice/Provisioning\ Profiles
|
mkdir -p ~/Library/MobileDevice/Provisioning\ Profiles
|
||||||
cp $PP_PATH ~/Library/MobileDevice/Provisioning\ Profiles
|
cp $PP_PATH ~/Library/MobileDevice/Provisioning\ Profiles
|
||||||
- uses: actions/setup-node@v6
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 16
|
node-version: 16
|
||||||
- run: npm install -g appdmg
|
- run: npm install -g appdmg
|
||||||
@@ -80,7 +78,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
python setup.py build_binary
|
python setup.py build_binary
|
||||||
- name: upload build
|
- name: upload build
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: mac-os-build-${{ runner.arch }}
|
name: mac-os-build-${{ runner.arch }}
|
||||||
path: dist/*.dmg
|
path: dist/*.dmg
|
||||||
@@ -89,8 +87,10 @@ jobs:
|
|||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
with:
|
with:
|
||||||
prerelease: true
|
prerelease: true
|
||||||
generate_release_notes: false
|
generate_release_notes: true
|
||||||
files: |
|
files: |
|
||||||
|
CHANGELOG.md
|
||||||
|
LICENSE.txt
|
||||||
dist/*.dmg
|
dist/*.dmg
|
||||||
- name: Clean up keychain and provisioning profile
|
- name: Clean up keychain and provisioning profile
|
||||||
# TODO signing
|
# TODO signing
|
||||||
|
|||||||
66
.github/workflows/package-osx-legacy.yml
vendored
66
.github/workflows/package-osx-legacy.yml
vendored
@@ -1,66 +0,0 @@
|
|||||||
name: build KCC for osx legacy
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "v*.*.*"
|
|
||||||
|
|
||||||
# Don't trigger if it's just a documentation update
|
|
||||||
paths-ignore:
|
|
||||||
- '**.md'
|
|
||||||
- '**.MD'
|
|
||||||
- '**.yml'
|
|
||||||
- '**.sh'
|
|
||||||
- 'docs/**'
|
|
||||||
- 'Dockerfile'
|
|
||||||
- 'LICENSE'
|
|
||||||
- '.gitattributes'
|
|
||||||
- '.gitignore'
|
|
||||||
- '.dockerignore'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
os: [ macos-15-intel ]
|
|
||||||
runs-on: ${{ matrix.os }}
|
|
||||||
env:
|
|
||||||
# We need the official Python, because the GA ones only support newer macOS versions
|
|
||||||
# The deployment target is picked up by the Python build tools automatically
|
|
||||||
PYTHON_VERSION: 3.11.9
|
|
||||||
MACOSX_DEPLOYMENT_TARGET: '10.14'
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v6
|
|
||||||
- name: Get Python
|
|
||||||
run: curl https://www.python.org/ftp/python/3.11.9/python-3.11.9-macos11.pkg -o "python.pkg"
|
|
||||||
- name: Install Python
|
|
||||||
run: |
|
|
||||||
sudo installer -pkg python.pkg -target /
|
|
||||||
- name: Install Python dependencies
|
|
||||||
run: |
|
|
||||||
python3 --version
|
|
||||||
pip3 install --upgrade pip setuptools wheel pyinstaller certifi
|
|
||||||
pip3 install --upgrade -r requirements-osx-legacy.txt
|
|
||||||
./gen_ui_files.sh
|
|
||||||
- uses: actions/setup-node@v6
|
|
||||||
with:
|
|
||||||
node-version: 16
|
|
||||||
- run: npm install -g appdmg
|
|
||||||
- name: build binary
|
|
||||||
run: |
|
|
||||||
python3 setup.py build_binary
|
|
||||||
- name: upload build
|
|
||||||
uses: actions/upload-artifact@v6
|
|
||||||
with:
|
|
||||||
name: osx-build-${{ runner.arch }}
|
|
||||||
path: dist/*.dmg
|
|
||||||
- name: Release
|
|
||||||
uses: softprops/action-gh-release@v2
|
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
|
||||||
with:
|
|
||||||
prerelease: true
|
|
||||||
generate_release_notes: false
|
|
||||||
files: |
|
|
||||||
LICENSE.txt
|
|
||||||
dist/*.dmg
|
|
||||||
57
.github/workflows/package-windows-with-docker.yml
vendored
Normal file
57
.github/workflows/package-windows-with-docker.yml
vendored
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
# For more information see: https://help.github.com/actions/language-and-framework-guides/using-python-with-github-actions
|
||||||
|
|
||||||
|
name: build KCC for windows with docker
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- "v*.*.*"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
# - name: Set up Python
|
||||||
|
# uses: actions/setup-python@v4
|
||||||
|
# with:
|
||||||
|
# python-version: 3.11
|
||||||
|
# cache: 'pip'
|
||||||
|
# - name: Install python dependencies
|
||||||
|
# run: |
|
||||||
|
# python -m pip install --upgrade pip setuptools wheel pyinstaller
|
||||||
|
# pip install -r requirements.txt
|
||||||
|
# - name: build binary
|
||||||
|
# run: |
|
||||||
|
# pyi-makespec -F -i icons\\comic2ebook.ico -n KCC_test -w --noupx kcc.py
|
||||||
|
- name: Package Application
|
||||||
|
uses: JackMcKew/pyinstaller-action-windows@main
|
||||||
|
with:
|
||||||
|
path: .
|
||||||
|
spec: ./kcc-c2e.spec
|
||||||
|
- name: Package Application
|
||||||
|
uses: JackMcKew/pyinstaller-action-windows@main
|
||||||
|
with:
|
||||||
|
path: .
|
||||||
|
spec: ./kcc-c2p.spec
|
||||||
|
- name: rename binaries
|
||||||
|
run: |
|
||||||
|
version_built=$(cat kindlecomicconverter/__init__.py | grep version | awk '{print $3}' | sed "s/[^.0-9b]//g")
|
||||||
|
mv dist/windows/kcc-c2e.exe dist/windows/KCC_c2e_${version_built}.exe
|
||||||
|
mv dist/windows/kcc-c2p.exe dist/windows/KCC_c2p_${version_built}.exe
|
||||||
|
- name: upload build
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: windows-build
|
||||||
|
path: dist/windows/*.exe
|
||||||
|
- name: Release
|
||||||
|
uses: softprops/action-gh-release@v2
|
||||||
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
|
with:
|
||||||
|
prerelease: true
|
||||||
|
generate_release_notes: true
|
||||||
|
files: |
|
||||||
|
CHANGELOG.md
|
||||||
|
LICENSE.txt
|
||||||
|
dist/windows/*.exe
|
||||||
40
.github/workflows/package-windows.yml
vendored
40
.github/workflows/package-windows.yml
vendored
@@ -23,21 +23,11 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
entry: [ kcc, kcc-c2e, kcc-c2p ]
|
|
||||||
include:
|
|
||||||
- entry: kcc
|
|
||||||
command: build_binary
|
|
||||||
- entry: kcc-c2e
|
|
||||||
command: build_c2e
|
|
||||||
- entry: kcc-c2p
|
|
||||||
command: build_c2p
|
|
||||||
runs-on: windows-latest
|
runs-on: windows-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v6
|
- uses: actions/checkout@v4
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v6
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: 3.11
|
python-version: 3.11
|
||||||
cache: 'pip'
|
cache: 'pip'
|
||||||
@@ -50,29 +40,19 @@ jobs:
|
|||||||
pip install certifi pyinstaller --no-binary pyinstaller
|
pip install certifi pyinstaller --no-binary pyinstaller
|
||||||
- name: build binary
|
- name: build binary
|
||||||
run: |
|
run: |
|
||||||
python setup.py ${{ matrix.command }}
|
python setup.py build_binary
|
||||||
- name: upload-unsigned-artifact
|
- name: upload build
|
||||||
id: upload-unsigned-artifact
|
uses: actions/upload-artifact@v4
|
||||||
uses: actions/upload-artifact@v6
|
|
||||||
with:
|
with:
|
||||||
name: windows-build-${{ matrix.entry }}
|
name: windows-build
|
||||||
path: dist/*.exe
|
path: dist/*.exe
|
||||||
- id: optional_step_id
|
|
||||||
uses: signpath/github-action-submit-signing-request@v2.0
|
|
||||||
if: ${{ github.repository == 'ciromattia/kcc' }}
|
|
||||||
with:
|
|
||||||
api-token: '${{ secrets.SIGNPATH_API_TOKEN }}'
|
|
||||||
organization-id: '1dc1bad6-4a8c-4f85-af30-5c5d3d392ea6'
|
|
||||||
project-slug: 'kcc'
|
|
||||||
signing-policy-slug: 'release-signing'
|
|
||||||
github-artifact-id: '${{ steps.upload-unsigned-artifact.outputs.artifact-id }}'
|
|
||||||
wait-for-completion: true
|
|
||||||
output-artifact-directory: 'dist/'
|
|
||||||
- name: Release
|
- name: Release
|
||||||
uses: softprops/action-gh-release@v2
|
uses: softprops/action-gh-release@v2
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
with:
|
with:
|
||||||
prerelease: true
|
prerelease: true
|
||||||
generate_release_notes: false
|
generate_release_notes: true
|
||||||
files: |
|
files: |
|
||||||
dist/*.exe
|
CHANGELOG.md
|
||||||
|
LICENSE.txt
|
||||||
|
dist/*.exe
|
||||||
60
.github/workflows/package-windows7.yml
vendored
60
.github/workflows/package-windows7.yml
vendored
@@ -1,60 +0,0 @@
|
|||||||
# For more information see: https://help.github.com/actions/language-and-framework-guides/using-python-with-github-actions
|
|
||||||
|
|
||||||
name: build KCC for windows 7
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "v*.*.*"
|
|
||||||
|
|
||||||
# Don't trigger if it's just a documentation update
|
|
||||||
paths-ignore:
|
|
||||||
- '**.md'
|
|
||||||
- '**.MD'
|
|
||||||
- '**.yml'
|
|
||||||
- '**.sh'
|
|
||||||
- 'docs/**'
|
|
||||||
- 'Dockerfile'
|
|
||||||
- 'LICENSE'
|
|
||||||
- '.gitattributes'
|
|
||||||
- '.gitignore'
|
|
||||||
- '.dockerignore'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: windows-2022
|
|
||||||
env:
|
|
||||||
WINDOWS_7: 1
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v6
|
|
||||||
- name: Set up Python
|
|
||||||
uses: actions/setup-python@v6
|
|
||||||
with:
|
|
||||||
python-version: 3.8
|
|
||||||
cache: 'pip'
|
|
||||||
- name: Install dependencies
|
|
||||||
env:
|
|
||||||
PYINSTALLER_COMPILE_BOOTLOADER: 1
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip setuptools wheel
|
|
||||||
pip install -r requirements-win7.txt
|
|
||||||
pip install certifi pyinstaller --no-binary pyinstaller
|
|
||||||
.\gen_ui_files.bat
|
|
||||||
- name: build binary
|
|
||||||
run: |
|
|
||||||
python setup.py build_binary
|
|
||||||
- name: upload-unsigned-artifact
|
|
||||||
id: upload-unsigned-artifact
|
|
||||||
uses: actions/upload-artifact@v6
|
|
||||||
with:
|
|
||||||
name: windows7-build
|
|
||||||
path: dist/*.exe
|
|
||||||
- name: Release
|
|
||||||
uses: softprops/action-gh-release@v2
|
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
|
||||||
with:
|
|
||||||
prerelease: true
|
|
||||||
generate_release_notes: false
|
|
||||||
files: |
|
|
||||||
dist/*.exe
|
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -8,9 +8,6 @@ dist/
|
|||||||
build/
|
build/
|
||||||
KindleComicConverter*.egg-info/
|
KindleComicConverter*.egg-info/
|
||||||
.idea/
|
.idea/
|
||||||
.vscode/
|
|
||||||
win7
|
|
||||||
osx10.11
|
|
||||||
/venv/
|
/venv/
|
||||||
/kindlegen*
|
/kindlegen*
|
||||||
/kcc.bat
|
/kcc.bat
|
||||||
|
|||||||
30
.travis.yml
Normal file
30
.travis.yml
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- os: osx
|
||||||
|
language: generic
|
||||||
|
osx_image: xcode11.1
|
||||||
|
|
||||||
|
before_install:
|
||||||
|
- pip3 install --upgrade pip setuptools wheel
|
||||||
|
|
||||||
|
install:
|
||||||
|
- pip3 install -r requirements.txt
|
||||||
|
- pip3 install certifi https://github.com/pyinstaller/pyinstaller/archive/develop.zip
|
||||||
|
- npm install -g appdmg
|
||||||
|
|
||||||
|
script: python3 setup.py build_binary
|
||||||
|
|
||||||
|
before_deploy:
|
||||||
|
- shopt -s extglob
|
||||||
|
- rm -r dist/!(*.deb|*.dmg)
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
provider: gcs
|
||||||
|
access_key_id: GOOG1EC62457RKUYFR2TIZUWV4EFSV2EP5LVLPPFXUAKADWJFDYPFW63BQSLA
|
||||||
|
secret_access_key:
|
||||||
|
secure: sxYjeho7U3im0Ezf6cz6TjYDiLvf0kAM2ETQHYoFNbD1VVvhJJyymDCnPH80zpFKmhc1MWTB6ndwsrPfcyZDLR2meSdWGPjZfFPY3RcrfImndKi7ln+mYQDBQ7W1lGit4YcH3Ju7LHceaTbRA7fVTX8pWKOcbXL2oM+lQxTJHH32+crVma+ChhbjzTWsSLRoakt3Nhiveec5p/qSW7AFe4Zq+b3C85IgwjSJI/xVwzaWrs6p915h1zZi7KL7YCMIxfQFrvRPFR2KTbh/DoLCCrqfbD4qh0PVy1li51Ac3hd/u3foiNnTNchzgE3Nv/nbKmtFU6huuLNgzkQGuLA+yn7mKYzBwA3ZmFgoimdH9+yRCMkZ8B5VHpvfN1hgpJcyEl1T98Kv4cdtRYNB4w9iAMy1qSVxhjeI+2rjuWGoXro0lU6L4LIRCOruY3AuLCAKG8Qw5Ak9ksmIKBhZ9soxpoIwu/TYDUQkFj29IrUQucg9TEp7uAoxu8/7EHxB7hWnBRaBAAQbMuIRg7yysT3FT0Os6SB0t9+RBsVMSPuIti9JJZ2Lu0uRI1+Se+g7ItzYtJoPhBJAzAa+J9OONj0RNj2z8Vq2oIBhH4z6b6zTRMVroos3cdfYl5qIKs9SQ7rmeHoPRROcqpCznsUZ/ESa4f2MewFU/7AYcEnCesZV4xg=
|
||||||
|
bucket: kcc-deploy
|
||||||
|
local-dir: dist
|
||||||
|
skip_cleanup: true
|
||||||
|
on:
|
||||||
|
repo: AcidWeb/KCC
|
||||||
90
Dockerfile
90
Dockerfile
@@ -1,77 +1,19 @@
|
|||||||
# STAGE 1: BUILDER
|
# Select final stage based on TARGETARCH ARG
|
||||||
# Contains all build tools and dev dependencies, will be discarded
|
FROM ghcr.io/ciromattia/kcc:docker-base-20241116
|
||||||
FROM python:3.13-slim-bullseye AS builder
|
LABEL com.kcc.name="Kindle Comic Converter"
|
||||||
|
LABEL com.kcc.author="Ciro Mattia Gonano, Paweł Jastrzębski and Darodi"
|
||||||
|
LABEL org.opencontainers.image.description='Kindle Comic Converter'
|
||||||
|
LABEL org.opencontainers.image.documentation='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.source='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.authors='darodi'
|
||||||
|
LABEL org.opencontainers.image.url='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.documentation='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.vendor='ciromattia'
|
||||||
|
LABEL org.opencontainers.image.licenses='ISC'
|
||||||
|
LABEL org.opencontainers.image.title="Kindle Comic Converter"
|
||||||
|
|
||||||
# Install system dependencies
|
COPY . /opt/kcc
|
||||||
RUN set -x && \
|
RUN cat /opt/kcc/kindlecomicconverter/__init__.py | grep version | awk '{print $3}' | sed "s/'//g" > /IMAGE_VERSION
|
||||||
BUILD_DEPS="build-essential cmake libffi-dev libfreetype6-dev libfontconfig1-dev libpng-dev libjpeg-dev libssl-dev libxft-dev make python3-dev python3-setuptools python3-wheel" && \
|
|
||||||
RUNTIME_DEPS="bash ca-certificates chrpath locales locales-all libfreetype6 libfontconfig1 p7zip-full python3 python3-pip libgl1" && \
|
|
||||||
DEBIAN_FRONTEND=noninteractive apt-get update -y && \
|
|
||||||
apt-get install -y --no-install-recommends ${BUILD_DEPS} ${RUNTIME_DEPS}
|
|
||||||
|
|
||||||
RUN \
|
ENTRYPOINT ["/opt/kcc/kcc-c2e.py"]
|
||||||
set -x && \
|
|
||||||
python -m venv /opt/venv && \
|
|
||||||
. /opt/venv/bin/activate && \
|
|
||||||
pip install --upgrade pip
|
|
||||||
|
|
||||||
# Install numpy first, as it is unlikely to change and takes too long to compile
|
|
||||||
RUN \
|
|
||||||
set -x && \
|
|
||||||
. /opt/venv/bin/activate && \
|
|
||||||
pip install --no-cache-dir numpy==2.3.4
|
|
||||||
|
|
||||||
# Install PyMuPDF separately, as it is likely to change but still takes too long to compile
|
|
||||||
RUN \
|
|
||||||
set -x && \
|
|
||||||
. /opt/venv/bin/activate && \
|
|
||||||
pip install --no-cache-dir PyMuPDF==1.26.6
|
|
||||||
|
|
||||||
# Install Python dependencies using virtual environment
|
|
||||||
COPY requirements-docker.txt .
|
|
||||||
|
|
||||||
RUN \
|
|
||||||
set -x && \
|
|
||||||
. /opt/venv/bin/activate && \
|
|
||||||
pip install --no-cache-dir -r requirements-docker.txt
|
|
||||||
|
|
||||||
# STAGE 2: FINAL
|
|
||||||
# Clean, small and secure image with only runtime dependencies
|
|
||||||
FROM python:3.13-slim-bullseye
|
|
||||||
|
|
||||||
# Install runtime dependencies only
|
|
||||||
RUN \
|
|
||||||
set -x && \
|
|
||||||
DEBIAN_FRONTEND=noninteractive apt-get update -y && \
|
|
||||||
apt-get install -y --no-install-recommends p7zip-full && \
|
|
||||||
rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
# Copy artifacts from builder
|
|
||||||
COPY --from=builder /opt/venv /opt/venv
|
|
||||||
COPY . /opt/kcc/
|
|
||||||
|
|
||||||
WORKDIR /opt/kcc
|
|
||||||
ENV PATH="/opt/venv/bin:$PATH"
|
|
||||||
|
|
||||||
# Setup executable and version file
|
|
||||||
RUN \
|
|
||||||
chmod +x /opt/kcc/entrypoint.sh && \
|
|
||||||
ln -s /opt/kcc/kcc-c2e.py /usr/local/bin/c2e && \
|
|
||||||
ln -s /opt/kcc/kcc-c2p.py /usr/local/bin/c2p && \
|
|
||||||
ln -s /opt/kcc/entrypoint.sh /usr/local/bin/entrypoint && \
|
|
||||||
ln -s /opt/kcc/kindlegen/kindlegen /usr/local/bin/kindlegen && \
|
|
||||||
cat /opt/kcc/kindlecomicconverter/__init__.py | grep version | awk '{print $3}' | sed "s/'//g" > /IMAGE_VERSION
|
|
||||||
|
|
||||||
LABEL com.kcc.name="Kindle Comic Converter" \
|
|
||||||
com.kcc.author="Ciro Mattia Gonano, Paweł Jastrzębski and Darodi" \
|
|
||||||
org.opencontainers.image.title="Kindle Comic Converter" \
|
|
||||||
org.opencontainers.image.description='Kindle Comic Converter' \
|
|
||||||
org.opencontainers.image.documentation='https://github.com/ciromattia/kcc' \
|
|
||||||
org.opencontainers.image.source='https://github.com/ciromattia/kcc' \
|
|
||||||
org.opencontainers.image.authors='Darodi and José Cerezo' \
|
|
||||||
org.opencontainers.image.url='https://github.com/ciromattia/kcc' \
|
|
||||||
org.opencontainers.image.vendor='ciromattia' \
|
|
||||||
org.opencontainers.image.licenses='ISC'
|
|
||||||
|
|
||||||
ENTRYPOINT ["entrypoint"]
|
|
||||||
CMD ["-h"]
|
CMD ["-h"]
|
||||||
|
|||||||
164
Dockerfile-base
Normal file
164
Dockerfile-base
Normal file
@@ -0,0 +1,164 @@
|
|||||||
|
FROM --platform=linux/amd64 python:3.13-slim-bullseye as compile-amd64
|
||||||
|
ARG TARGETOS
|
||||||
|
ARG TARGETARCH
|
||||||
|
ARG TARGETVARIANT
|
||||||
|
RUN echo "I'm building for $TARGETOS/$TARGETARCH/$TARGETVARIANT"
|
||||||
|
|
||||||
|
|
||||||
|
COPY requirements.txt /opt/kcc/
|
||||||
|
ENV PATH="/opt/venv/bin:$PATH"
|
||||||
|
RUN DEBIAN_FRONTEND=noninteractive apt-get update -y && apt-get -yq upgrade && \
|
||||||
|
apt-get install -y libpng-dev libjpeg-dev p7zip-full unrar-free libgl1 && \
|
||||||
|
python -m pip install --upgrade pip && \
|
||||||
|
python -m venv /opt/venv && \
|
||||||
|
python -m pip install -r /opt/kcc/requirements.txt
|
||||||
|
|
||||||
|
|
||||||
|
######################################################################################
|
||||||
|
|
||||||
|
FROM --platform=linux/arm64 python:3.13-slim-bullseye as compile-arm64
|
||||||
|
ARG TARGETOS
|
||||||
|
ARG TARGETARCH
|
||||||
|
ARG TARGETVARIANT
|
||||||
|
RUN echo "I'm building for $TARGETOS/$TARGETARCH/$TARGETVARIANT"
|
||||||
|
|
||||||
|
ENV LC_ALL=C.UTF-8 \
|
||||||
|
LANG=C.UTF-8 \
|
||||||
|
LANGUAGE=en_US:en
|
||||||
|
|
||||||
|
SHELL ["/bin/bash", "-o", "pipefail", "-c"]
|
||||||
|
|
||||||
|
COPY requirements.txt /opt/kcc/
|
||||||
|
ENV PATH="/opt/venv/bin:$PATH"
|
||||||
|
|
||||||
|
RUN set -x && \
|
||||||
|
TEMP_PACKAGES=() && \
|
||||||
|
KEPT_PACKAGES=() && \
|
||||||
|
# Packages only required during build
|
||||||
|
TEMP_PACKAGES+=(build-essential) && \
|
||||||
|
TEMP_PACKAGES+=(cmake) && \
|
||||||
|
TEMP_PACKAGES+=(libfreetype6-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libfontconfig1-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libpng-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libjpeg-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libssl-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libxft-dev) && \
|
||||||
|
TEMP_PACKAGES+=(make) && \
|
||||||
|
TEMP_PACKAGES+=(python3-dev) && \
|
||||||
|
TEMP_PACKAGES+=(python3-setuptools) && \
|
||||||
|
TEMP_PACKAGES+=(python3-wheel) && \
|
||||||
|
# Packages kept in the image
|
||||||
|
KEPT_PACKAGES+=(bash) && \
|
||||||
|
KEPT_PACKAGES+=(ca-certificates) && \
|
||||||
|
KEPT_PACKAGES+=(chrpath) && \
|
||||||
|
KEPT_PACKAGES+=(locales) && \
|
||||||
|
KEPT_PACKAGES+=(locales-all) && \
|
||||||
|
KEPT_PACKAGES+=(libfreetype6) && \
|
||||||
|
KEPT_PACKAGES+=(libfontconfig1) && \
|
||||||
|
KEPT_PACKAGES+=(p7zip-full) && \
|
||||||
|
KEPT_PACKAGES+=(python3) && \
|
||||||
|
KEPT_PACKAGES+=(python3-pip) && \
|
||||||
|
KEPT_PACKAGES+=(unrar-free) && \
|
||||||
|
# Install packages
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get update -y && apt-get -yq upgrade && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
${KEPT_PACKAGES[@]} \
|
||||||
|
${TEMP_PACKAGES[@]} \
|
||||||
|
&& \
|
||||||
|
# Install required python modules
|
||||||
|
python -m pip install --upgrade pip && \
|
||||||
|
python -m venv /opt/venv && \
|
||||||
|
python -m pip install -r /opt/kcc/requirements.txt
|
||||||
|
|
||||||
|
|
||||||
|
######################################################################################
|
||||||
|
|
||||||
|
FROM --platform=linux/arm/v7 python:3.13-slim-bullseye as compile-armv7
|
||||||
|
ARG TARGETOS
|
||||||
|
ARG TARGETARCH
|
||||||
|
ARG TARGETVARIANT
|
||||||
|
RUN echo "I'm building for $TARGETOS/$TARGETARCH/$TARGETVARIANT"
|
||||||
|
|
||||||
|
ENV LC_ALL=C.UTF-8 \
|
||||||
|
LANG=C.UTF-8 \
|
||||||
|
LANGUAGE=en_US:en
|
||||||
|
|
||||||
|
SHELL ["/bin/bash", "-o", "pipefail", "-c"]
|
||||||
|
|
||||||
|
COPY requirements.txt /opt/kcc/
|
||||||
|
ENV PATH="/opt/venv/bin:$PATH"
|
||||||
|
|
||||||
|
RUN set -x && \
|
||||||
|
TEMP_PACKAGES=() && \
|
||||||
|
KEPT_PACKAGES=() && \
|
||||||
|
# Packages only required during build
|
||||||
|
TEMP_PACKAGES+=(build-essential) && \
|
||||||
|
TEMP_PACKAGES+=(cmake) && \
|
||||||
|
TEMP_PACKAGES+=(libffi-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libfreetype6-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libfontconfig1-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libpng-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libjpeg-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libssl-dev) && \
|
||||||
|
TEMP_PACKAGES+=(libxft-dev) && \
|
||||||
|
TEMP_PACKAGES+=(make) && \
|
||||||
|
TEMP_PACKAGES+=(python3-dev) && \
|
||||||
|
TEMP_PACKAGES+=(python3-setuptools) && \
|
||||||
|
TEMP_PACKAGES+=(python3-wheel) && \
|
||||||
|
# Packages kept in the image
|
||||||
|
KEPT_PACKAGES+=(bash) && \
|
||||||
|
KEPT_PACKAGES+=(ca-certificates) && \
|
||||||
|
KEPT_PACKAGES+=(chrpath) && \
|
||||||
|
KEPT_PACKAGES+=(locales) && \
|
||||||
|
KEPT_PACKAGES+=(locales-all) && \
|
||||||
|
KEPT_PACKAGES+=(libfreetype6) && \
|
||||||
|
KEPT_PACKAGES+=(libfontconfig1) && \
|
||||||
|
KEPT_PACKAGES+=(p7zip-full) && \
|
||||||
|
KEPT_PACKAGES+=(python3) && \
|
||||||
|
KEPT_PACKAGES+=(python3-pip) && \
|
||||||
|
KEPT_PACKAGES+=(unrar-free) && \
|
||||||
|
# Install packages
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get update -y && apt-get -yq upgrade && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
${KEPT_PACKAGES[@]} \
|
||||||
|
${TEMP_PACKAGES[@]} \
|
||||||
|
&& \
|
||||||
|
# Install required python modules
|
||||||
|
python -m pip install --upgrade pip && \
|
||||||
|
python -m venv /opt/venv && \
|
||||||
|
python -m pip install --upgrade pillow psutil requests python-slugify raven packaging mozjpeg-lossless-optimization natsort distro numpy
|
||||||
|
|
||||||
|
|
||||||
|
######################################################################################
|
||||||
|
FROM --platform=linux/amd64 python:3.13-slim-bullseye as build-amd64
|
||||||
|
COPY --from=compile-amd64 /opt/venv /opt/venv
|
||||||
|
|
||||||
|
FROM --platform=linux/arm64 python:3.13-slim-bullseye as build-arm64
|
||||||
|
COPY --from=compile-arm64 /opt/venv /opt/venv
|
||||||
|
|
||||||
|
FROM --platform=linux/arm/v7 python:3.13-slim-bullseye as build-armv7
|
||||||
|
COPY --from=compile-armv7 /opt/venv /opt/venv
|
||||||
|
######################################################################################
|
||||||
|
|
||||||
|
# Select final stage based on TARGETARCH ARG
|
||||||
|
FROM build-${TARGETARCH}${TARGETVARIANT}
|
||||||
|
LABEL com.kcc.name="Kindle Comic Converter base image"
|
||||||
|
LABEL com.kcc.author="Ciro Mattia Gonano, Paweł Jastrzębski and Darodi"
|
||||||
|
LABEL org.opencontainers.image.description='Kindle Comic Converter base image'
|
||||||
|
LABEL org.opencontainers.image.documentation='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.source='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.authors='darodi'
|
||||||
|
LABEL org.opencontainers.image.url='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.documentation='https://github.com/ciromattia/kcc'
|
||||||
|
LABEL org.opencontainers.image.vendor='ciromattia'
|
||||||
|
LABEL org.opencontainers.image.licenses='ISC'
|
||||||
|
LABEL org.opencontainers.image.title="Kindle Comic Converter"
|
||||||
|
|
||||||
|
|
||||||
|
ENV PATH="/opt/venv/bin:$PATH"
|
||||||
|
WORKDIR /app
|
||||||
|
RUN DEBIAN_FRONTEND=noninteractive apt-get update -y && apt-get -yq upgrade && \
|
||||||
|
apt-get install -y p7zip-full unrar-free && \
|
||||||
|
ln -s /app/kindlegen /bin/kindlegen && \
|
||||||
|
echo docker-base-20241116 > /IMAGE_VERSION
|
||||||
|
|
||||||
@@ -1,9 +1,8 @@
|
|||||||
ISC LICENSE
|
ISC LICENSE
|
||||||
|
|
||||||
Copyright (c) 2012-2025 Ciro Mattia Gonano <ciromattia@gmail.com>
|
Copyright (c) 2012-2014 Ciro Mattia Gonano <ciromattia@gmail.com>
|
||||||
Copyright (c) 2013-2019 Paweł Jastrzębski <pawelj@iosphe.re>
|
Copyright (c) 2013-2019 Paweł Jastrzębski <pawelj@iosphe.re>
|
||||||
Copyright (c) 2021-2023 Darodi (https://github.com/darodi)
|
Copyright (c) 2021-2023 Darodi
|
||||||
Copyright (c) 2023-2025 Alex Xu (https://github.com/axu2)
|
|
||||||
|
|
||||||
Permission to use, copy, modify, and/or distribute this software for
|
Permission to use, copy, modify, and/or distribute this software for
|
||||||
any purpose with or without fee is hereby granted, provided that the
|
any purpose with or without fee is hereby granted, provided that the
|
||||||
|
|||||||
1
MANIFEST.in
Normal file
1
MANIFEST.in
Normal file
@@ -0,0 +1 @@
|
|||||||
|
exclude kindlecomicconverter/sentry.py
|
||||||
254
README.md
254
README.md
@@ -1,62 +1,15 @@
|
|||||||
<img src="header.jpg" alt="Header Image" width="400">
|
|
||||||
|
|
||||||
# KCC
|
# KCC
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
[](https://github.com/ciromattia/kcc/releases)
|
[](https://github.com/ciromattia/kcc/releases)
|
||||||
[](https://github.com/ciromattia/kcc/pkgs/container/kcc)
|
[](https://github.com/ciromattia/kcc/pkgs/container/kcc)
|
||||||
[](https://github.com/ciromattia/kcc/releases)
|
|
||||||
|
|
||||||
|
|
||||||
**Kindle Comic Converter** optimizes black & white (or color) comics and manga for E-ink ereaders
|
**Kindle Comic Converter** is a Python app to convert comic/manga files or folders to EPUB, Panel View MOBI or E-Ink optimized CBZ.
|
||||||
like Kindle, Kobo, ReMarkable, and more.
|
It was initially developed for Kindle but since version 4.6 it outputs valid EPUB 3.0 so _**despite its name, KCC is
|
||||||
Pages display in fullscreen without margins,
|
actually a comic/manga to EPUB converter that every e-reader owner can happily use**_.
|
||||||
with proper fixed layout support.
|
It can also optionally optimize images by applying a number of transformations.
|
||||||
Supported input formats include JPG/PNG image files in folders, archives, or PDFs.
|
|
||||||
Supported output formats include MOBI/AZW3, EPUB, KEPUB, CBZ, and PDF.
|
|
||||||
KCC runs on Windows, macOS, and Linux.
|
|
||||||
|
|
||||||
Just drop your input files into the KCC window, hit convert, and USB drop the output files onto your device's `documents` folder!
|
|
||||||
|
|
||||||
https://github.com/user-attachments/assets/da73d625-e082-482d-91a4-ae4765e96fd7
|
|
||||||
|
|
||||||
**WARNING**: Kindle Scribe 2025 support may not be possible. Does not work well currently.
|
|
||||||
|
|
||||||
**NEW**: PDF output is now supported for direct conversion to reMarkable devices!
|
|
||||||
When using a reMarkable profile (Rmk1, Rmk2, RmkPP), the format automatically defaults to PDF
|
|
||||||
for optimal compatibility with your device's native PDF reader.
|
|
||||||
|
|
||||||
The absolute highest quality source files are print quality DRM-free PDFs from Kodansha/[Humble Bundle](https://humblebundleinc.sjv.io/xL6Zv1)/Fanatical,
|
|
||||||
which can be directly converted by KCC.
|
|
||||||
|
|
||||||
Its main feature is various optional image processing steps to look good on eink screens,
|
|
||||||
which have different requirements than normal LCD screens.
|
|
||||||
Combining that with downscaling to your specific device's screen resolution
|
|
||||||
can result in filesize reductions of hundreds of MB per volume with no visible quality loss on eink.
|
|
||||||
This can also improve battery life, page turn speed, and general performance
|
|
||||||
on underpowered ereaders with small memory and storage capacities.
|
|
||||||
|
|
||||||
KCC avoids many common formatting issues (some of which occur [even on the Kindle Store](https://github.com/ciromattia/kcc/wiki/Kindle-Store-bad-formatting)), such as:
|
|
||||||
1) faded black levels causing unneccessarily low contrast, which is hard to see and can cause eyestrain.
|
|
||||||
2) unneccessary margins at the bottom of the screen
|
|
||||||
3) Not utilizing the full 1860x2480 resolution of the 10" Kindle Scribe
|
|
||||||
4) incorrect page turn direction for manga that's read right to left
|
|
||||||
5) unaligned two page spreads in landscape, where pages are shifted over by 1
|
|
||||||
6) Removing without blur the rainbow effect on color eink Kaleido 3 due to manga screentones
|
|
||||||
|
|
||||||
The GUI looks like this, built in Qt6, with my most commonly used settings:
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
Simply drag and drop your files/folders into the KCC window,
|
|
||||||
adjust your settings (hover over each option to see details in a tooltip),
|
|
||||||
and hit convert to create ereader optimized files.
|
|
||||||
You can change the default output directory by holding `Shift` while clicking the convert button.
|
|
||||||
Then just drag and drop the generated output files onto your device's documents folder via USB.
|
|
||||||
If you are on macOS and use a 2022+ Kindle, you may need to use Amazon USB File Manager for Mac.
|
|
||||||
|
|
||||||
YouTube tutorial (please subscribe): https://www.youtube.com/watch?v=QQ6zJcMF2Iw
|
|
||||||
|
|
||||||
Installation tutorial: https://www.youtube.com/watch?v=IR2Fhcm9658
|
|
||||||
|
|
||||||
### A word of warning
|
### A word of warning
|
||||||
**KCC** _is not_ [Amazon's Kindle Comic Creator](http://www.amazon.com/gp/feature.html?ie=UTF8&docId=1001103761) nor is in any way endorsed by Amazon.
|
**KCC** _is not_ [Amazon's Kindle Comic Creator](http://www.amazon.com/gp/feature.html?ie=UTF8&docId=1001103761) nor is in any way endorsed by Amazon.
|
||||||
@@ -69,29 +22,15 @@ If you have some **technical** problems using KCC please [file an issue here](ht
|
|||||||
If you can fix an open issue, fork & make a pull request.
|
If you can fix an open issue, fork & make a pull request.
|
||||||
|
|
||||||
If you find **KCC** valuable you can consider donating to the authors:
|
If you find **KCC** valuable you can consider donating to the authors:
|
||||||
- Ciro Mattia Gonano (founder, active 2012-2014):
|
- Ciro Mattia Gonano (founder, active 2013-2014):
|
||||||
|
- [](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=D8WNYNPBGDAS2)
|
||||||
[](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=D8WNYNPBGDAS2)
|
- [](http://flattr.com/thing/2260449/ciromattiakcc-on-GitHub)
|
||||||
|
|
||||||
- Paweł Jastrzębski (active 2013-2019):
|
- Paweł Jastrzębski (active 2013-2019):
|
||||||
|
- [](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=YTTJ4LK2JDHPS)
|
||||||
[](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=YTTJ4LK2JDHPS)
|
- [](https://jastrzeb.ski/donate/)
|
||||||
[](https://jastrzeb.ski/donate/)
|
|
||||||
|
|
||||||
- Alex Xu (active 2023-Present)
|
- Alex Xu (active 2023-Present)
|
||||||
|
- [](https://www.paypal.com/donate/?business=QFJVE7A6LCP6U&no_recurring=0&item_name=Kindle+Comic+Converter¤cy_code=USD)
|
||||||
|
|
||||||
[](https://ko-fi.com/Q5Q41BW8HS)
|
|
||||||
|
|
||||||
## Commissions
|
|
||||||
|
|
||||||
This section is subject to change:
|
|
||||||
|
|
||||||
Email (for commisions and inquiries): `kindle.comic.converter` gmail
|
|
||||||
|
|
||||||
|
|
||||||
## Sponsors
|
|
||||||
|
|
||||||
- Free code signing on Windows provided by [SignPath.io](https://about.signpath.io/), certificate by [SignPath Foundation](https://signpath.org/)
|
|
||||||
|
|
||||||
## DOWNLOADS
|
## DOWNLOADS
|
||||||
|
|
||||||
@@ -102,46 +41,20 @@ Click on **Assets** of the latest release.
|
|||||||
You probably want either
|
You probably want either
|
||||||
- `KCC_*.*.*.exe` (Windows)
|
- `KCC_*.*.*.exe` (Windows)
|
||||||
- `kcc_macos_arm_*.*.*.dmg` (recent Mac with Apple Silicon M1 chip or later)
|
- `kcc_macos_arm_*.*.*.dmg` (recent Mac with Apple Silicon M1 chip or later)
|
||||||
- `kcc_macos_i386_*.*.*.dmg` (older Mac with Intel chip macOS 14+)
|
- `kcc_macos_i386_*.*.*.dmg` (older Mac with Intel chip)
|
||||||
|
|
||||||
There are also legacy macOS 10.14+ and Windows 7 experimental versions available.
|
|
||||||
|
|
||||||
The `c2e` and `c2p` versions are command line tools for power users.
|
The `c2e` and `c2p` versions are command line tools for power users.
|
||||||
|
|
||||||
On macOS, if you get a `can't be opened` error, follow: https://support.apple.com/guide/mac-help/open-a-mac-app-from-an-unknown-developer-mh40616/mac
|
On Windows 11, you may need to run in compatibility mode for an older Windows version.
|
||||||
|
|
||||||
|
On Mac, right click open to get past the security warning.
|
||||||
|
|
||||||
For flatpak, Docker, and AppImage versions, refer to the wiki: https://github.com/ciromattia/kcc/wiki/Installation
|
For flatpak, Docker, and AppImage versions, refer to the wiki: https://github.com/ciromattia/kcc/wiki/Installation
|
||||||
|
|
||||||
## FAQ
|
## FAQ
|
||||||
- Should I use Calibre?
|
|
||||||
- No. Calibre doesn't properly support fixed layout EPUB/MOBI, so modifying KCC output (even just metadata!) in Calibre can break the formatting.
|
|
||||||
Viewing KCC output in Calibre will also not work properly.
|
|
||||||
On 7th gen and later Kindles running firmware 5.15.1+, you can get cover thumbnails simply by USB dropping into documents folder.
|
|
||||||
On 6th gen and older, you can get cover thumbnails by keeping Kindle plugged in during conversion.
|
|
||||||
If you are careful to not modify the file however, you can still use Calibre, but direct USB dropping is reccomended.
|
|
||||||
- Blank pages?
|
|
||||||
- May happen when [using PNG with Kindle Scribe](https://github.com/ciromattia/kcc/issues/665) or [any format with a Kindle Colorsoft](https://github.com/ciromattia/kcc/issues/768). Solve by using JPG with Kindle Scribe or buying a Kobo Colour. Happens more often when turning pages really fast.
|
|
||||||
Going back a few pages and exiting and re-entering book should fix it temporarily.
|
|
||||||
- What output format should I use?
|
|
||||||
- MOBI for Kindles. CBZ for Kindle DX. CBZ for Koreader. KEPUB for Kobo. PDF for ReMarkable.
|
|
||||||
- All options have additional information in tooltips if you hover over the option.
|
|
||||||
- To get the converted book onto your Kindle/Kobo, just drag and drop the mobi/kepub into the documents folder on your Kindle/Kobo via USB
|
|
||||||
- Kindle panel view not working?
|
|
||||||
- Virtual panel view is enabled in Aa menu on your Kindle, not in KCC as of 7.4
|
|
||||||
- Right to left mode not working?
|
|
||||||
- RTL mode only affects splitting order for CBZ output. Your cbz reader itself sets the page turn direction.
|
|
||||||
- Colors inverted?
|
|
||||||
- Disable Kindle dark mode
|
|
||||||
- Cannot connect Kindle Scribe or 2024+ Kindle to macOS
|
|
||||||
- Use official MTP [Amazon USB File Transfer app](https://www.amazon.com/gp/help/customer/display.html/ref=hp_Connect_USB_MTP?nodeId=TCUBEdEkbIhK07ysFu)
|
|
||||||
(no login required). Works much better than previously recommended Android File Transfer. Cannot run simutaneously with other transfer apps.
|
|
||||||
- How to make AZW3 instead of MOBI?
|
|
||||||
- The `.mobi` file generated by KCC is a dual filetype, it's both MOBI and AZW3. The file extension is `.mobi` for compatibility reasons.
|
|
||||||
- [Windows 7 support](https://github.com/ciromattia/kcc/issues/678)
|
- [Windows 7 support](https://github.com/ciromattia/kcc/issues/678)
|
||||||
- Image too dark?
|
- [Combine files/chapters](https://github.com/ciromattia/kcc/issues/612#issuecomment-2117985011)
|
||||||
- The default gamma correction of 1.8 makes the image darker, and is useful for faded/gray artwork/text. Disable by setting gamma = 1.0
|
- [Flatpak mobi conversion stuck](https://github.com/ciromattia/kcc/wiki/Installation#linux)
|
||||||
- Huge margins / slow page turns?
|
|
||||||
- You likely modified the file during transfer using a 3rd party app. Try simply dragging and dropping the final mobi/kepub file into the Kindle documents folder via USB.
|
|
||||||
|
|
||||||
## PREREQUISITES
|
## PREREQUISITES
|
||||||
|
|
||||||
@@ -155,11 +68,9 @@ If you have issues detecting it, get stuck on the MOBI conversion step, or use L
|
|||||||
|
|
||||||
### 7-Zip
|
### 7-Zip
|
||||||
|
|
||||||
This is optional but will make conversions much faster.
|
This is only required for certain files and advanced features.
|
||||||
|
|
||||||
This is required for certain files and advanced features.
|
KCC will ask you to install if needed.
|
||||||
|
|
||||||
KCC will ask you to install if needed.
|
|
||||||
|
|
||||||
Refer to the wiki to install: https://github.com/ciromattia/kcc/wiki/Installation#7-zip
|
Refer to the wiki to install: https://github.com/ciromattia/kcc/wiki/Installation#7-zip
|
||||||
|
|
||||||
@@ -187,44 +98,36 @@ sudo apt-get install python3 p7zip-full python3-pil python3-psutil python3-slugi
|
|||||||
### Profiles:
|
### Profiles:
|
||||||
|
|
||||||
```
|
```
|
||||||
'K1': ("Kindle 1", (600, 670), Palette4, 1.0),
|
'K1': ("Kindle 1", (600, 670), Palette4, 1.8),
|
||||||
'K2': ("Kindle 2", (600, 670), Palette15, 1.0),
|
'K11': ("Kindle 11", (1072, 1448), Palette16, 1.8),
|
||||||
'K11': ("Kindle 11", (1072, 1448), Palette16, 1.0),
|
'K2': ("Kindle 2", (600, 670), Palette15, 1.8),
|
||||||
'K34': ("Kindle Keyboard/Touch", (600, 800), Palette16, 1.0),
|
'K34': ("Kindle Keyboard/Touch", (600, 800), Palette16, 1.8),
|
||||||
'K57': ("Kindle 5/7", (600, 800), Palette16, 1.0),
|
'K578': ("Kindle", (600, 800), Palette16, 1.8),
|
||||||
'K810': ("Kindle 8/10", (600, 800), Palette16, 1.0),
|
'KDX': ("Kindle DX/DXG", (824, 1000), Palette16, 1.8),
|
||||||
'KDX': ("Kindle DX/DXG", (824, 1000), Palette16, 1.0),
|
'KPW': ("Kindle Paperwhite 1/2", (758, 1024), Palette16, 1.8),
|
||||||
'KPW': ("Kindle Paperwhite 1/2", (758, 1024), Palette16, 1.0),
|
'KV': ("Kindle Paperwhite 3/4/Voyage/Oasis", (1072, 1448), Palette16, 1.8),
|
||||||
'KV': ("Kindle Voyage", (1072, 1448), Palette16, 1.0),
|
'KPW5': ("Kindle Paperwhite 5/Signature Edition", (1236, 1648), Palette16, 1.8),
|
||||||
'KPW34': ("Kindle Paperwhite 3/4", (1072, 1448), Palette16, 1.0),
|
'KO': ("Kindle Oasis 2/3/Paperwhite 12/Colorsoft 12", (1264, 1680), Palette16, 1.8),
|
||||||
'KPW5': ("Kindle Paperwhite 5/Signature Edition", (1236, 1648), Palette16, 1.0),
|
'KS': ("Kindle Scribe", (1860, 2480), Palette16, 1.8),
|
||||||
'KO': ("Kindle Oasis 2/3/Paperwhite 12", (1264, 1680), Palette16, 1.0),
|
'KoMT': ("Kobo Mini/Touch", (600, 800), Palette16, 1.8),
|
||||||
'KCS': ("Kindle Colorsoft", (1264, 1680), Palette16, 1.0),
|
'KoG': ("Kobo Glo", (768, 1024), Palette16, 1.8),
|
||||||
'KS1860': ("Kindle 1860", (1860, 1920), Palette16, 1.0),
|
'KoGHD': ("Kobo Glo HD", (1072, 1448), Palette16, 1.8),
|
||||||
'KS1920': ("Kindle 1920", (1920, 1920), Palette16, 1.0),
|
'KoA': ("Kobo Aura", (758, 1024), Palette16, 1.8),
|
||||||
'KS': ("Kindle Scribe 1/2", (1860, 2480), Palette16, 1.0),
|
'KoAHD': ("Kobo Aura HD", (1080, 1440), Palette16, 1.8),
|
||||||
'KS3': ("Kindle Scribe 3", (1986, 2648), Palette16, 1.0),
|
'KoAH2O': ("Kobo Aura H2O", (1080, 1430), Palette16, 1.8),
|
||||||
'KSCS': ("Kindle Scribe Colorsoft", (1986, 2648), Palette16, 1.0),
|
'KoAO': ("Kobo Aura ONE", (1404, 1872), Palette16, 1.8),
|
||||||
'KoMT': ("Kobo Mini/Touch", (600, 800), Palette16, 1.0),
|
'KoN': ("Kobo Nia", (758, 1024), Palette16, 1.8),
|
||||||
'KoG': ("Kobo Glo", (768, 1024), Palette16, 1.0),
|
'KoC': ("Kobo Clara HD/Kobo Clara 2E", (1072, 1448), Palette16, 1.8),
|
||||||
'KoGHD': ("Kobo Glo HD", (1072, 1448), Palette16, 1.0),
|
'KoCC': ("Kobo Clara Colour", (1072, 1448), Palette16, 1.8),
|
||||||
'KoA': ("Kobo Aura", (758, 1024), Palette16, 1.0),
|
'KoL': ("Kobo Libra H2O/Kobo Libra 2", (1264, 1680), Palette16, 1.8),
|
||||||
'KoAHD': ("Kobo Aura HD", (1080, 1440), Palette16, 1.0),
|
'KoLC': ("Kobo Libra Colour", (1264, 1680), Palette16, 1.8),
|
||||||
'KoAH2O': ("Kobo Aura H2O", (1080, 1430), Palette16, 1.0),
|
'KoF': ("Kobo Forma", (1440, 1920), Palette16, 1.8),
|
||||||
'KoAO': ("Kobo Aura ONE", (1404, 1872), Palette16, 1.0),
|
'KoS': ("Kobo Sage", (1440, 1920), Palette16, 1.8),
|
||||||
'KoN': ("Kobo Nia", (758, 1024), Palette16, 1.0),
|
'KoE': ("Kobo Elipsa", (1404, 1872), Palette16, 1.8),
|
||||||
'KoC': ("Kobo Clara HD/Kobo Clara 2E", (1072, 1448), Palette16, 1.0),
|
'Rmk1': ("reMarkable 1", (1404, 1872), Palette16, 1.8),
|
||||||
'KoCC': ("Kobo Clara Colour", (1072, 1448), Palette16, 1.0),
|
'Rmk2': ("reMarkable 2", (1404, 1872), Palette16, 1.8),
|
||||||
'KoL': ("Kobo Libra H2O/Kobo Libra 2", (1264, 1680), Palette16, 1.0),
|
'RmkPP': ("reMarkable Paper Pro", (1620, 2160), Palette16, 1.8),
|
||||||
'KoLC': ("Kobo Libra Colour", (1264, 1680), Palette16, 1.0),
|
'OTHER': ("Other", (0, 0), Palette16, 1.8),
|
||||||
'KoF': ("Kobo Forma", (1440, 1920), Palette16, 1.0),
|
|
||||||
'KoS': ("Kobo Sage", (1440, 1920), Palette16, 1.0),
|
|
||||||
'KoE': ("Kobo Elipsa", (1404, 1872), Palette16, 1.0),
|
|
||||||
'Rmk1': ("reMarkable 1", (1404, 1872), Palette16, 1.0),
|
|
||||||
'Rmk2': ("reMarkable 2", (1404, 1872), Palette16, 1.0),
|
|
||||||
'RmkPP': ("reMarkable Paper Pro", (1620, 2160), Palette16, 1.0),
|
|
||||||
'RmkPPMove': ("reMarkable Paper Pro Move", (954, 1696), Palette16, 1.0),
|
|
||||||
'OTHER': ("Other", (0, 0), Palette16, 1.0),
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Standalone `kcc-c2e.py` usage:
|
### Standalone `kcc-c2e.py` usage:
|
||||||
@@ -247,33 +150,24 @@ MAIN:
|
|||||||
the maximal size of output file in MB. [Default=100MB for webtoon and 400MB for others]
|
the maximal size of output file in MB. [Default=100MB for webtoon and 400MB for others]
|
||||||
|
|
||||||
PROCESSING:
|
PROCESSING:
|
||||||
-n, --noprocessing Do not modify image and ignore any profile or processing option
|
-n, --noprocessing Do not modify image and ignore any profil or processing option
|
||||||
--pdfextract Use legacy PDF image extraction method from KCC 8 and earlier.
|
|
||||||
-u, --upscale Resize images smaller than device's resolution
|
-u, --upscale Resize images smaller than device's resolution
|
||||||
-s, --stretch Stretch images to device's resolution
|
-s, --stretch Stretch images to device's resolution
|
||||||
-r SPLITTER, --splitter SPLITTER
|
-r SPLITTER, --splitter SPLITTER
|
||||||
Double page parsing mode. 0: Split 1: Rotate 2: Both [Default=0]
|
Double page parsing mode. 0: Split 1: Rotate 2: Both [Default=0]
|
||||||
-g GAMMA, --gamma GAMMA
|
-g GAMMA, --gamma GAMMA
|
||||||
Apply gamma correction to linearize the image [Default=Auto]
|
Apply gamma correction to linearize the image [Default=Auto]
|
||||||
--autolevel Set most common dark pixel value to be black point for leveling.
|
|
||||||
--noautocontrast Disable autocontrast
|
|
||||||
--colorautocontrast Force autocontrast for all pages. Skipped when near blacks and whites don't exist
|
|
||||||
-c CROPPING, --cropping CROPPING
|
-c CROPPING, --cropping CROPPING
|
||||||
Set cropping mode. 0: Disabled 1: Margins 2: Margins + page numbers [Default=2]
|
Set cropping mode. 0: Disabled 1: Margins 2: Margins + page numbers [Default=2]
|
||||||
--cp CROPPINGP, --croppingpower CROPPINGP
|
--cp CROPPINGP, --croppingpower CROPPINGP
|
||||||
Set cropping power [Default=1.0]
|
Set cropping power [Default=1.0]
|
||||||
--preservemargin After calculating crop, "back up" a specified percentage amount [Default=0]
|
|
||||||
--cm CROPPINGM, --croppingminimum CROPPINGM
|
--cm CROPPINGM, --croppingminimum CROPPINGM
|
||||||
Set cropping minimum area ratio [Default=0.0]
|
Set cropping minimum area ratio [Default=0.0]
|
||||||
--ipc INTERPANELCROP, --interpanelcrop INTERPANELCROP
|
|
||||||
Crop empty sections. 0: Disabled 1: Horizontally 2: Both [Default=0]
|
|
||||||
--blackborders Disable autodetection and force black borders
|
--blackborders Disable autodetection and force black borders
|
||||||
--whiteborders Disable autodetection and force white borders
|
--whiteborders Disable autodetection and force white borders
|
||||||
--coverfill Center-crop only the cover to fill target device screen
|
|
||||||
--forcecolor Don't convert images to grayscale
|
--forcecolor Don't convert images to grayscale
|
||||||
--forcepng Create PNG files instead JPEG
|
--forcepng Create PNG files instead JPEG
|
||||||
--mozjpeg Create JPEG files using mozJpeg
|
--mozjpeg Create JPEG files using mozJpeg
|
||||||
--jpeg-quality The JPEG quality, on a scale from 0 (worst) to 95 (best). Default 85 for most devices.
|
|
||||||
--maximizestrips Turn 1x4 strips to 2x2 strips
|
--maximizestrips Turn 1x4 strips to 2x2 strips
|
||||||
-d, --delete Delete source file(s) or a directory. It's not recoverable.
|
-d, --delete Delete source file(s) or a directory. It's not recoverable.
|
||||||
|
|
||||||
@@ -282,19 +176,15 @@ OUTPUT SETTINGS:
|
|||||||
Output generated file to specified directory or file
|
Output generated file to specified directory or file
|
||||||
-t TITLE, --title TITLE
|
-t TITLE, --title TITLE
|
||||||
Comic title [Default=filename or directory name]
|
Comic title [Default=filename or directory name]
|
||||||
--metadatatitle Write title using ComicInfo.xml or other embedded metadata. 0: Don't use Title from metadata 1: Combine Title with default schema 2: Use Title only [Default=0]
|
|
||||||
-a AUTHOR, --author AUTHOR
|
-a AUTHOR, --author AUTHOR
|
||||||
Author name [Default=KCC]
|
Author name [Default=KCC]
|
||||||
-f FORMAT, --format FORMAT
|
-f FORMAT, --format FORMAT
|
||||||
Output format (Available options: Auto, MOBI, EPUB, CBZ, PDF, KFX, MOBI+EPUB) [Default=Auto]
|
Output format (Available options: Auto, MOBI, EPUB, CBZ, KFX, MOBI+EPUB) [Default=Auto]
|
||||||
--nokepub If format is EPUB, output file with '.epub' extension rather than '.kepub.epub'
|
--nokepub If format is EPUB, output file with '.epub' extension rather than '.kepub.epub'
|
||||||
-b BATCHSPLIT, --batchsplit BATCHSPLIT
|
-b BATCHSPLIT, --batchsplit BATCHSPLIT
|
||||||
Split output into multiple files. 0: Don't split 1: Automatic mode 2: Consider every subdirectory as separate volume [Default=0]
|
Split output into multiple files. 0: Don't split 1: Automatic mode 2: Consider every subdirectory as separate volume [Default=0]
|
||||||
--spreadshift Shift first page to opposite side in landscape for two page spread alignment
|
--spreadshift Shift first page to opposite side in landscape for two page spread alignment
|
||||||
--norotate Do not rotate double page spreads in spread splitter option.
|
--norotate Do not rotate double page spreads in spread splitter option.
|
||||||
--rotatefirst Put rotated spread first in spread splitter option.
|
|
||||||
--filefusion Combines all input files into a single file.
|
|
||||||
--eraserainbow Erase rainbow effect on color eink screen by attenuating interfering frequencies
|
|
||||||
|
|
||||||
CUSTOM PROFILE:
|
CUSTOM PROFILE:
|
||||||
--customwidth CUSTOMWIDTH
|
--customwidth CUSTOMWIDTH
|
||||||
@@ -330,26 +220,17 @@ OTHER:
|
|||||||
|
|
||||||
This section is for developers who want to contribute to KCC or power users who want to run the latest code without waiting for an official release.
|
This section is for developers who want to contribute to KCC or power users who want to run the latest code without waiting for an official release.
|
||||||
|
|
||||||
Easiest to use [GitHub Desktop](https://desktop.github.com) to clone your fork of the KCC repo. From GitHub Desktop, click on `Repository` in the toolbar, then `Command Prompt` (Windows)/`Terminal` (Mac) to open a window in the KCC repo.
|
Easiest to use [GitHub Desktop](https://desktop.github.com) to clone the KCC repo. From GitHub Desktop, click on `Repository` in the toolbar, then `Command Prompt` (Windows)/`Terminal` (Mac) to open a window in the KCC repo.
|
||||||
|
|
||||||
Depending on your system [Python](https://www.python.org) may be called either `python` or `python3`. We use virtual environments (venv) to manage dependencies.
|
Depending on your system [Python](https://www.python.org) may be called either `python` or `python3`. We use virtual environments (venv) to manage dependencies.
|
||||||
|
|
||||||
If you want to edit the code, a good code editor is [VS Code](https://code.visualstudio.com).
|
If you want to edit the code, a good code editor is [VS Code](https://code.visualstudio.com).
|
||||||
|
|
||||||
If you want to edit the `.ui` files, use `pyside6-designer` which is included in the `pip install pyside6`.
|
If you want to edit the `.ui` files, use [Qt Creator](https://www.qt.io/download-qt-installer-oss), included in **Qt for desktop development**.
|
||||||
If new objects have been added, verify that correct tab order has been applied by using [Tab Order Editing Mode](https://doc.qt.io/qt-6/designer-tab-order.html).
|
|
||||||
Then use the `gen_ui_files` scripts to autogenerate the python UI.
|
Then use the `gen_ui_files` scripts to autogenerate the python UI.
|
||||||
|
|
||||||
An example PR adding a new checkbox is here: https://github.com/ciromattia/kcc/pull/785
|
An example PR adding a new checkbox is here: https://github.com/ciromattia/kcc/pull/785
|
||||||
|
|
||||||
video of adding a new checkbox: https://youtu.be/g3I8DU74C7g
|
|
||||||
|
|
||||||
Do not use `git merge` to merge master from upstream,
|
|
||||||
use the "Sync fork" button on your fork on GitHub in your branch
|
|
||||||
to avoid weird looking merges in pull requests.
|
|
||||||
|
|
||||||
When making changes, be aware of how your change might affect file splitting/chunking
|
|
||||||
or chapter alignment.
|
|
||||||
|
|
||||||
### Windows install from source
|
### Windows install from source
|
||||||
|
|
||||||
@@ -368,16 +249,8 @@ venv\Scripts\activate.bat
|
|||||||
python kcc.py
|
python kcc.py
|
||||||
```
|
```
|
||||||
|
|
||||||
You can build a `.exe` of KCC like the downloads we offer with
|
|
||||||
|
|
||||||
```
|
|
||||||
python setup.py build_binary
|
|
||||||
```
|
|
||||||
|
|
||||||
### macOS install from source
|
### macOS install from source
|
||||||
|
|
||||||
If the system installed Python gives you issues, please install the latest Python from either brew or the official website.
|
|
||||||
|
|
||||||
One time setup and running for the first time:
|
One time setup and running for the first time:
|
||||||
```
|
```
|
||||||
python3 -m venv venv
|
python3 -m venv venv
|
||||||
@@ -393,12 +266,6 @@ source venv/bin/activate
|
|||||||
python kcc.py
|
python kcc.py
|
||||||
```
|
```
|
||||||
|
|
||||||
You can build a `.app` of KCC like the downloads we offer with
|
|
||||||
|
|
||||||
```
|
|
||||||
python setup.py build_binary
|
|
||||||
```
|
|
||||||
|
|
||||||
## CREDITS
|
## CREDITS
|
||||||
**KCC** is made by
|
**KCC** is made by
|
||||||
|
|
||||||
@@ -416,12 +283,6 @@ The app relies and includes the following scripts:
|
|||||||
- Icon is by **Nikolay Verin** ([http://ncrow.deviantart.com/](http://ncrow.deviantart.com/)) and released under [CC BY-NC-SA 3.0](http://creativecommons.org/licenses/by-nc-sa/3.0/) License.
|
- Icon is by **Nikolay Verin** ([http://ncrow.deviantart.com/](http://ncrow.deviantart.com/)) and released under [CC BY-NC-SA 3.0](http://creativecommons.org/licenses/by-nc-sa/3.0/) License.
|
||||||
|
|
||||||
## SAMPLE FILES CREATED BY KCC
|
## SAMPLE FILES CREATED BY KCC
|
||||||
|
|
||||||
https://www.mediafire.com/folder/ixh40veo6hrc5/kcc_samples
|
|
||||||
|
|
||||||
Older links (dead):
|
|
||||||
|
|
||||||
|
|
||||||
* [Kindle Oasis 2 / 3](http://kcc.iosphe.re/Samples/Ubunchu!-KO.mobi)
|
* [Kindle Oasis 2 / 3](http://kcc.iosphe.re/Samples/Ubunchu!-KO.mobi)
|
||||||
* [Kindle Paperwhite 3 / 4 / Voyage / Oasis](http://kcc.iosphe.re/Samples/Ubunchu!-KV.mobi)
|
* [Kindle Paperwhite 3 / 4 / Voyage / Oasis](http://kcc.iosphe.re/Samples/Ubunchu!-KV.mobi)
|
||||||
* [Kindle Paperwhite 1 / 2](http://kcc.iosphe.re/Samples/Ubunchu!-KPW.mobi)
|
* [Kindle Paperwhite 1 / 2](http://kcc.iosphe.re/Samples/Ubunchu!-KPW.mobi)
|
||||||
@@ -434,15 +295,12 @@ Older links (dead):
|
|||||||
|
|
||||||
## PRIVACY
|
## PRIVACY
|
||||||
**KCC** is initiating internet connections in two cases:
|
**KCC** is initiating internet connections in two cases:
|
||||||
* During startup - Version check and announcement check.
|
* During startup - Version check.
|
||||||
* When error occurs - Automatic reporting on Windows and macOS.
|
* When error occurs - Automatic reporting on Windows and macOS.
|
||||||
|
|
||||||
## KNOWN ISSUES
|
## KNOWN ISSUES
|
||||||
Please check [wiki page](https://github.com/ciromattia/kcc/wiki/Known-issues).
|
Please check [wiki page](https://github.com/ciromattia/kcc/wiki/Known-issues).
|
||||||
|
|
||||||
## COPYRIGHT
|
## COPYRIGHT
|
||||||
Copyright (c) 2012-2025 Ciro Mattia Gonano, Paweł Jastrzębski, Darodi and Alex Xu.
|
Copyright (c) 2012-2023 Ciro Mattia Gonano, Paweł Jastrzębski and Darodi.
|
||||||
**KCC** is released under ISC LICENSE; see [LICENSE.txt](./LICENSE.txt) for further details.
|
**KCC** is released under ISC LICENSE; see [LICENSE.txt](./LICENSE.txt) for further details.
|
||||||
|
|
||||||
## Verification
|
|
||||||
Impact-Site-Verification: ffe48fc7-4f0c-40fd-bd2e-59f4d7205180
|
|
||||||
|
|||||||
14
appveyor.yml
Normal file
14
appveyor.yml
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
environment:
|
||||||
|
PYTHON: "C:\\Python37-x64"
|
||||||
|
|
||||||
|
install:
|
||||||
|
- set PATH="%PYTHON%\\Scripts";%PATH%
|
||||||
|
- "%PYTHON%\\python.exe -m pip install --upgrade pip setuptools wheel"
|
||||||
|
- "%PYTHON%\\python.exe -m pip install -r requirements.txt"
|
||||||
|
- "%PYTHON%\\python.exe -m pip install certifi https://github.com/pyinstaller/pyinstaller/archive/develop.zip"
|
||||||
|
|
||||||
|
build_script:
|
||||||
|
- "%PYTHON%\\python.exe setup.py build_binary"
|
||||||
|
|
||||||
|
artifacts:
|
||||||
|
- path: dist\KCC*
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
MODE=${KCC_MODE:-c2e}
|
|
||||||
|
|
||||||
case "$MODE" in
|
|
||||||
"c2e")
|
|
||||||
echo "Starting C2E..."
|
|
||||||
exec c2e "$@"
|
|
||||||
;;
|
|
||||||
|
|
||||||
"c2p")
|
|
||||||
echo "Starting C2P..."
|
|
||||||
exec c2p "$@"
|
|
||||||
;;
|
|
||||||
|
|
||||||
*)
|
|
||||||
echo "Error: Unknown mode '$MODE'" >&2
|
|
||||||
exit 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
16
environment.yml
Normal file
16
environment.yml
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
name: kcc
|
||||||
|
channels:
|
||||||
|
- conda-forge
|
||||||
|
- defaults
|
||||||
|
dependencies:
|
||||||
|
- python=3.11
|
||||||
|
- Pillow>=5.2.0
|
||||||
|
- psutil>=5.9.5
|
||||||
|
- python-slugify>=1.2.1
|
||||||
|
- raven>=6.0.0
|
||||||
|
- distro
|
||||||
|
- natsort>=8.4.0
|
||||||
|
- pip
|
||||||
|
- pip:
|
||||||
|
- mozjpeg-lossless-optimization>=1.1.2
|
||||||
|
- pyside6>=6.5.1
|
||||||
@@ -28,9 +28,4 @@
|
|||||||
<file>../icons/document_new.png</file>
|
<file>../icons/document_new.png</file>
|
||||||
<file>../icons/folder_new.png</file>
|
<file>../icons/folder_new.png</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
<qresource prefix="Brand">
|
|
||||||
<file>../icons/kofi_symbol.png</file>
|
|
||||||
<file>../icons/Humble_H-Red.png</file>
|
|
||||||
<file>../icons/Bindle_Red.png</file>
|
|
||||||
</qresource>
|
|
||||||
</RCC>
|
</RCC>
|
||||||
|
|||||||
1179
gui/KCC.ui
1179
gui/KCC.ui
File diff suppressed because it is too large
Load Diff
@@ -62,66 +62,56 @@
|
|||||||
<item row="1" column="1">
|
<item row="1" column="1">
|
||||||
<widget class="QLineEdit" name="volumeLine"/>
|
<widget class="QLineEdit" name="volumeLine"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="3" column="0">
|
<item row="2" column="0">
|
||||||
<widget class="QLabel" name="label_3">
|
<widget class="QLabel" name="label_3">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Number:</string>
|
<string>Number:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="3" column="1">
|
<item row="2" column="1">
|
||||||
<widget class="QLineEdit" name="numberLine"/>
|
<widget class="QLineEdit" name="numberLine"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="4" column="0">
|
<item row="3" column="0">
|
||||||
<widget class="QLabel" name="label_4">
|
<widget class="QLabel" name="label_4">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Writer:</string>
|
<string>Writer:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="4" column="1">
|
<item row="3" column="1">
|
||||||
<widget class="QLineEdit" name="writerLine"/>
|
<widget class="QLineEdit" name="writerLine"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="5" column="0">
|
<item row="4" column="0">
|
||||||
<widget class="QLabel" name="label_5">
|
<widget class="QLabel" name="label_5">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Penciller:</string>
|
<string>Penciller:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="5" column="1">
|
<item row="4" column="1">
|
||||||
<widget class="QLineEdit" name="pencillerLine"/>
|
<widget class="QLineEdit" name="pencillerLine"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="6" column="0">
|
<item row="5" column="0">
|
||||||
<widget class="QLabel" name="label_6">
|
<widget class="QLabel" name="label_6">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Inker:</string>
|
<string>Inker:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="6" column="1">
|
<item row="5" column="1">
|
||||||
<widget class="QLineEdit" name="inkerLine"/>
|
<widget class="QLineEdit" name="inkerLine"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="7" column="0">
|
<item row="6" column="0">
|
||||||
<widget class="QLabel" name="label_7">
|
<widget class="QLabel" name="label_7">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Colorist:</string>
|
<string>Colorist:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item row="7" column="1">
|
<item row="6" column="1">
|
||||||
<widget class="QLineEdit" name="coloristLine"/>
|
<widget class="QLineEdit" name="coloristLine"/>
|
||||||
</item>
|
</item>
|
||||||
<item row="2" column="0">
|
|
||||||
<widget class="QLabel" name="label_8">
|
|
||||||
<property name="text">
|
|
||||||
<string>Title:</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item row="2" column="1">
|
|
||||||
<widget class="QLineEdit" name="titleLine"/>
|
|
||||||
</item>
|
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
@@ -192,18 +182,6 @@
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
<tabstops>
|
|
||||||
<tabstop>seriesLine</tabstop>
|
|
||||||
<tabstop>volumeLine</tabstop>
|
|
||||||
<tabstop>titleLine</tabstop>
|
|
||||||
<tabstop>numberLine</tabstop>
|
|
||||||
<tabstop>writerLine</tabstop>
|
|
||||||
<tabstop>pencillerLine</tabstop>
|
|
||||||
<tabstop>inkerLine</tabstop>
|
|
||||||
<tabstop>coloristLine</tabstop>
|
|
||||||
<tabstop>okButton</tabstop>
|
|
||||||
<tabstop>cancelButton</tabstop>
|
|
||||||
</tabstops>
|
|
||||||
<resources>
|
<resources>
|
||||||
<include location="KCC.qrc"/>
|
<include location="KCC.qrc"/>
|
||||||
</resources>
|
</resources>
|
||||||
|
|||||||
BIN
header.jpg
BIN
header.jpg
Binary file not shown.
|
Before Width: | Height: | Size: 921 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 8.1 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 16 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 9.1 KiB |
@@ -11,7 +11,7 @@ a = Analysis(['kcc-c2e.py'],
|
|||||||
hiddenimports=['_cffi_backend'],
|
hiddenimports=['_cffi_backend'],
|
||||||
hookspath=[],
|
hookspath=[],
|
||||||
runtime_hooks=[],
|
runtime_hooks=[],
|
||||||
excludes=['pkg_resources'],
|
excludes=[],
|
||||||
win_no_prefer_redirects=False,
|
win_no_prefer_redirects=False,
|
||||||
win_private_assemblies=False,
|
win_private_assemblies=False,
|
||||||
cipher=block_cipher,
|
cipher=block_cipher,
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ a = Analysis(['kcc-c2p.py'],
|
|||||||
hiddenimports=['_cffi_backend'],
|
hiddenimports=['_cffi_backend'],
|
||||||
hookspath=[],
|
hookspath=[],
|
||||||
runtime_hooks=[],
|
runtime_hooks=[],
|
||||||
excludes=['pkg_resources'],
|
excludes=[],
|
||||||
win_no_prefer_redirects=False,
|
win_no_prefer_redirects=False,
|
||||||
win_private_assemblies=False,
|
win_private_assemblies=False,
|
||||||
cipher=block_cipher,
|
cipher=block_cipher,
|
||||||
|
|||||||
39
kcc.spec
39
kcc.spec
@@ -1,39 +0,0 @@
|
|||||||
# -*- mode: python ; coding: utf-8 -*-
|
|
||||||
|
|
||||||
|
|
||||||
block_cipher = None
|
|
||||||
|
|
||||||
|
|
||||||
a = Analysis(['kcc.py'],
|
|
||||||
pathex=['.'],
|
|
||||||
binaries=[],
|
|
||||||
datas=[],
|
|
||||||
hiddenimports=['_cffi_backend'],
|
|
||||||
hookspath=[],
|
|
||||||
runtime_hooks=[],
|
|
||||||
excludes=['pkg_resources'],
|
|
||||||
win_no_prefer_redirects=False,
|
|
||||||
win_private_assemblies=False,
|
|
||||||
cipher=block_cipher,
|
|
||||||
noarchive=False)
|
|
||||||
pyz = PYZ(a.pure, a.zipped_data,
|
|
||||||
cipher=block_cipher)
|
|
||||||
|
|
||||||
exe = EXE(pyz,
|
|
||||||
a.scripts,
|
|
||||||
a.binaries,
|
|
||||||
a.zipfiles,
|
|
||||||
a.datas,
|
|
||||||
[],
|
|
||||||
name='kcc',
|
|
||||||
debug=False,
|
|
||||||
bootloader_ignore_signals=False,
|
|
||||||
strip=False,
|
|
||||||
upx=False,
|
|
||||||
upx_exclude=[],
|
|
||||||
runtime_tmpdir=None,
|
|
||||||
console=False,
|
|
||||||
disable_windowed_traceback=False,
|
|
||||||
target_arch=None,
|
|
||||||
codesign_identity=None,
|
|
||||||
entitlements_file=None , icon='icons\\comic2ebook.ico')
|
|
||||||
@@ -16,13 +16,9 @@
|
|||||||
# OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
# OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
||||||
# TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
# TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
||||||
# PERFORMANCE OF THIS SOFTWARE.
|
# PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
|
||||||
from datetime import datetime, timezone
|
|
||||||
import itertools
|
|
||||||
from pathlib import Path
|
|
||||||
from PySide6.QtCore import (QSize, QUrl, Qt, Signal, QIODeviceBase, QEvent, QThread, QSettings)
|
from PySide6.QtCore import (QSize, QUrl, Qt, Signal, QIODeviceBase, QEvent, QThread, QSettings)
|
||||||
from PySide6.QtGui import (QColor, QIcon, QPixmap, QDesktopServices)
|
from PySide6.QtGui import (QColor, QIcon, QPixmap, QDesktopServices)
|
||||||
from PySide6.QtWidgets import (QApplication, QLabel, QListWidgetItem, QMainWindow, QSystemTrayIcon, QFileDialog, QMessageBox, QDialog, QTreeView, QAbstractItemView)
|
from PySide6.QtWidgets import (QApplication, QLabel, QListWidgetItem, QMainWindow, QApplication, QSystemTrayIcon, QFileDialog, QMessageBox, QDialog)
|
||||||
from PySide6.QtNetwork import (QLocalSocket, QLocalServer)
|
from PySide6.QtNetwork import (QLocalSocket, QLocalServer)
|
||||||
|
|
||||||
import os
|
import os
|
||||||
@@ -31,7 +27,7 @@ import sys
|
|||||||
from urllib.parse import unquote
|
from urllib.parse import unquote
|
||||||
from time import sleep
|
from time import sleep
|
||||||
from shutil import move, rmtree
|
from shutil import move, rmtree
|
||||||
from subprocess import STDOUT, PIPE, CalledProcessError
|
from subprocess import STDOUT, PIPE
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
from xml.sax.saxutils import escape
|
from xml.sax.saxutils import escape
|
||||||
@@ -42,7 +38,6 @@ from raven import Client
|
|||||||
from tempfile import gettempdir
|
from tempfile import gettempdir
|
||||||
|
|
||||||
from .shared import HTMLStripper, sanitizeTrace, walkLevel, subprocess_run
|
from .shared import HTMLStripper, sanitizeTrace, walkLevel, subprocess_run
|
||||||
from .comicarchive import SEVENZIP, TAR, available_archive_tools
|
|
||||||
from . import __version__
|
from . import __version__
|
||||||
from . import comic2ebook
|
from . import comic2ebook
|
||||||
from . import metadata
|
from . import metadata
|
||||||
@@ -124,8 +119,6 @@ class Icons:
|
|||||||
self.CBZFormat.addPixmap(QPixmap(":/Formats/icons/CBZ.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
self.CBZFormat.addPixmap(QPixmap(":/Formats/icons/CBZ.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
||||||
self.EPUBFormat = QIcon()
|
self.EPUBFormat = QIcon()
|
||||||
self.EPUBFormat.addPixmap(QPixmap(":/Formats/icons/EPUB.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
self.EPUBFormat.addPixmap(QPixmap(":/Formats/icons/EPUB.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
||||||
self.KFXFormat = QIcon()
|
|
||||||
self.KFXFormat.addPixmap(QPixmap(":/Formats/icons/KFX.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
|
||||||
|
|
||||||
self.info = QIcon()
|
self.info = QIcon()
|
||||||
self.info.addPixmap(QPixmap(":/Status/icons/info.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
self.info.addPixmap(QPixmap(":/Status/icons/info.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
||||||
@@ -137,15 +130,6 @@ class Icons:
|
|||||||
self.programIcon = QIcon()
|
self.programIcon = QIcon()
|
||||||
self.programIcon.addPixmap(QPixmap(":/Icon/icons/comic2ebook.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
self.programIcon.addPixmap(QPixmap(":/Icon/icons/comic2ebook.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
||||||
|
|
||||||
self.kofi = QIcon()
|
|
||||||
self.kofi.addPixmap(QPixmap(":/Brand/icons/kofi_symbol.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
|
||||||
|
|
||||||
self.humble = QIcon()
|
|
||||||
self.humble.addPixmap(QPixmap(":/Brand/icons/Humble_H-Red.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
|
||||||
|
|
||||||
self.bindle = QIcon()
|
|
||||||
self.bindle.addPixmap(QPixmap(":/Brand/icons/Bindle_Red.png"), QIcon.Mode.Normal, QIcon.State.Off)
|
|
||||||
|
|
||||||
|
|
||||||
class VersionThread(QThread):
|
class VersionThread(QThread):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
@@ -160,52 +144,19 @@ class VersionThread(QThread):
|
|||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
try:
|
try:
|
||||||
# unauthenticated API requests limit is 60 req/hour
|
json_parser = requests.get("https://api.github.com/repos/ciromattia/kcc/releases/latest").json()
|
||||||
if getattr(sys, 'frozen', False):
|
|
||||||
json_parser = requests.get("https://api.github.com/repos/ciromattia/kcc/releases/latest").json()
|
|
||||||
|
|
||||||
html_url = json_parser["html_url"]
|
html_url = json_parser["html_url"]
|
||||||
latest_version = json_parser["tag_name"]
|
latest_version = json_parser["tag_name"]
|
||||||
latest_version = re.sub(r'^v', "", latest_version)
|
latest_version = re.sub(r'^v', "", latest_version)
|
||||||
|
|
||||||
if ("b" not in __version__ and Version(latest_version) > Version(__version__)) \
|
if ("b" not in __version__ and Version(latest_version) > Version(__version__)) \
|
||||||
or ("b" in __version__
|
or ("b" in __version__
|
||||||
and Version(latest_version) >= Version(re.sub(r'b.*', '', __version__))):
|
and Version(latest_version) >= Version(re.sub(r'b.*', '', __version__))):
|
||||||
MW.addMessage.emit('<a href="' + html_url + '"><b>The new version is available!</b></a>', 'warning',
|
MW.addMessage.emit('<a href="' + html_url + '"><b>The new version is available!</b></a>', 'warning',
|
||||||
False)
|
False)
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
return
|
||||||
|
|
||||||
try:
|
|
||||||
announcements = requests.get('https://api.github.com/repos/axu2/kcc-messages/contents/links.json',
|
|
||||||
headers={
|
|
||||||
'Accept': 'application/vnd.github.raw+json',
|
|
||||||
'X-GitHub-Api-Version': '2022-11-28'}).json()
|
|
||||||
for category, payloads in announcements.items():
|
|
||||||
for payload in payloads:
|
|
||||||
expiration = datetime.fromisoformat(payload['expiration'])
|
|
||||||
if expiration < datetime.now(timezone.utc):
|
|
||||||
continue
|
|
||||||
delta = expiration - datetime.now(timezone.utc)
|
|
||||||
time_left = f"{delta.days} day(s) left"
|
|
||||||
icon = 'info'
|
|
||||||
if category == 'humbleMangaBundles':
|
|
||||||
icon = 'humble'
|
|
||||||
if category == 'humbleComicBundles':
|
|
||||||
icon = 'bindle'
|
|
||||||
if category == 'kofi':
|
|
||||||
icon = 'kofi'
|
|
||||||
message = f"<b>{payload.get('name')}</b>"
|
|
||||||
if payload.get('link'):
|
|
||||||
message = '<a href="{}"><b>{}</b></a>'.format(payload.get('link'), payload.get('name'))
|
|
||||||
if payload.get('showDeadline'):
|
|
||||||
message += f': {time_left}'
|
|
||||||
if category == 'humbleBundles':
|
|
||||||
message += ' [referral]'
|
|
||||||
MW.addMessage.emit(message, icon , False)
|
|
||||||
except Exception as e:
|
|
||||||
print(e)
|
|
||||||
|
|
||||||
|
|
||||||
def setAnswer(self, dialoganswer):
|
def setAnswer(self, dialoganswer):
|
||||||
self.answer = dialoganswer
|
self.answer = dialoganswer
|
||||||
@@ -290,29 +241,9 @@ class WorkerThread(QThread):
|
|||||||
options.upscale = True
|
options.upscale = True
|
||||||
if GUI.gammaBox.isChecked() and float(GUI.gammaValue) > 0.09:
|
if GUI.gammaBox.isChecked() and float(GUI.gammaValue) > 0.09:
|
||||||
options.gamma = float(GUI.gammaValue)
|
options.gamma = float(GUI.gammaValue)
|
||||||
if GUI.autoLevelBox.isChecked():
|
options.cropping = GUI.croppingBox.checkState().value
|
||||||
options.autolevel = True
|
|
||||||
if GUI.autocontrastBox.checkState() == Qt.CheckState.PartiallyChecked:
|
|
||||||
options.noautocontrast = True
|
|
||||||
elif GUI.autocontrastBox.checkState() == Qt.CheckState.Checked:
|
|
||||||
options.colorautocontrast = True
|
|
||||||
if GUI.croppingBox.isChecked():
|
|
||||||
if GUI.croppingBox.checkState() == Qt.CheckState.PartiallyChecked:
|
|
||||||
options.cropping = 1
|
|
||||||
else:
|
|
||||||
options.cropping = 2
|
|
||||||
else:
|
|
||||||
options.cropping = 0
|
|
||||||
if GUI.croppingBox.checkState() != Qt.CheckState.Unchecked:
|
if GUI.croppingBox.checkState() != Qt.CheckState.Unchecked:
|
||||||
options.croppingp = float(GUI.croppingPowerValue)
|
options.croppingp = float(GUI.croppingPowerValue)
|
||||||
options.preservemargin = GUI.preserveMarginBox.value()
|
|
||||||
if GUI.interPanelCropBox.isChecked():
|
|
||||||
if GUI.interPanelCropBox.checkState() == Qt.CheckState.PartiallyChecked:
|
|
||||||
options.interpanelcrop = 1
|
|
||||||
else:
|
|
||||||
options.interpanelcrop = 2
|
|
||||||
else:
|
|
||||||
options.interpanelcrop = 0
|
|
||||||
if GUI.borderBox.checkState() == Qt.CheckState.PartiallyChecked:
|
if GUI.borderBox.checkState() == Qt.CheckState.PartiallyChecked:
|
||||||
options.white_borders = True
|
options.white_borders = True
|
||||||
elif GUI.borderBox.checkState() == Qt.CheckState.Checked:
|
elif GUI.borderBox.checkState() == Qt.CheckState.Checked:
|
||||||
@@ -321,87 +252,43 @@ class WorkerThread(QThread):
|
|||||||
options.batchsplit = 2
|
options.batchsplit = 2
|
||||||
if GUI.colorBox.isChecked():
|
if GUI.colorBox.isChecked():
|
||||||
options.forcecolor = True
|
options.forcecolor = True
|
||||||
if GUI.eraseRainbowBox.isChecked():
|
|
||||||
options.eraserainbow = True
|
|
||||||
if GUI.maximizeStrips.isChecked():
|
if GUI.maximizeStrips.isChecked():
|
||||||
options.maximizestrips = True
|
options.maximizestrips = True
|
||||||
if GUI.disableProcessingBox.isChecked():
|
if GUI.disableProcessingBox.isChecked():
|
||||||
options.noprocessing = True
|
options.noprocessing = True
|
||||||
if GUI.pdfExtractBox.isChecked():
|
|
||||||
options.pdfextract = True
|
|
||||||
if GUI.coverFillBox.isChecked():
|
|
||||||
options.coverfill = True
|
|
||||||
if GUI.metadataTitleBox.checkState() == Qt.CheckState.PartiallyChecked:
|
|
||||||
options.metadatatitle = 1
|
|
||||||
elif GUI.metadataTitleBox.checkState() == Qt.CheckState.Checked:
|
|
||||||
options.metadatatitle = 2
|
|
||||||
if GUI.deleteBox.isChecked():
|
if GUI.deleteBox.isChecked():
|
||||||
options.delete = True
|
options.delete = True
|
||||||
if GUI.spreadShiftBox.isChecked():
|
if GUI.spreadShiftBox.isChecked():
|
||||||
options.spreadshift = True
|
options.spreadshift = True
|
||||||
if GUI.fileFusionBox.isChecked():
|
|
||||||
options.filefusion = True
|
|
||||||
else:
|
|
||||||
options.filefusion = False
|
|
||||||
if GUI.noRotateBox.isChecked():
|
if GUI.noRotateBox.isChecked():
|
||||||
options.norotate = True
|
options.norotate = True
|
||||||
if GUI.rotateFirstBox.isChecked():
|
|
||||||
options.rotatefirst = True
|
|
||||||
if GUI.mozJpegBox.checkState() == Qt.CheckState.PartiallyChecked:
|
if GUI.mozJpegBox.checkState() == Qt.CheckState.PartiallyChecked:
|
||||||
options.forcepng = True
|
options.forcepng = True
|
||||||
elif GUI.mozJpegBox.checkState() == Qt.CheckState.Checked:
|
elif GUI.mozJpegBox.checkState() == Qt.CheckState.Checked:
|
||||||
options.mozjpeg = True
|
options.mozjpeg = True
|
||||||
if GUI.jpegQualityBox.isChecked():
|
|
||||||
options.jpegquality = GUI.jpegQualitySpinBox.value()
|
|
||||||
if GUI.currentMode > 2:
|
if GUI.currentMode > 2:
|
||||||
options.customwidth = str(GUI.widthBox.value())
|
options.customwidth = str(GUI.widthBox.value())
|
||||||
options.customheight = str(GUI.heightBox.value())
|
options.customheight = str(GUI.heightBox.value())
|
||||||
if GUI.targetDirectory != '':
|
if GUI.targetDirectory != '':
|
||||||
options.output = GUI.targetDirectory
|
options.output = GUI.targetDirectory
|
||||||
if GUI.titleEdit.text():
|
|
||||||
options.title = str(GUI.titleEdit.text())
|
|
||||||
if GUI.authorEdit.text():
|
if GUI.authorEdit.text():
|
||||||
options.author = str(GUI.authorEdit.text())
|
options.author = str(GUI.authorEdit.text())
|
||||||
if GUI.chunkSizeCheckBox.isChecked():
|
|
||||||
options.targetsize = int(GUI.chunkSizeBox.value())
|
|
||||||
|
|
||||||
for i in range(GUI.jobList.count()):
|
for i in range(GUI.jobList.count()):
|
||||||
# Make sure that we don't consider any system message as job to do
|
# Make sure that we don't consider any system message as job to do
|
||||||
if GUI.jobList.item(i).icon().isNull():
|
if GUI.jobList.item(i).icon().isNull():
|
||||||
currentJobs.append(str(GUI.jobList.item(i).text()))
|
currentJobs.append(str(GUI.jobList.item(i).text()))
|
||||||
GUI.jobList.clear()
|
GUI.jobList.clear()
|
||||||
if options.filefusion:
|
for job in currentJobs:
|
||||||
bookDir = []
|
|
||||||
MW.addMessage.emit('Attempting file fusion', 'info', False)
|
|
||||||
for job in currentJobs:
|
|
||||||
bookDir.append(job)
|
|
||||||
try:
|
|
||||||
comic2ebook.options = comic2ebook.checkOptions(copy(options))
|
|
||||||
currentJobs.clear()
|
|
||||||
currentJobs.append(comic2ebook.makeFusion(bookDir))
|
|
||||||
MW.addMessage.emit('Created fusion at ' + currentJobs[0], 'info', False)
|
|
||||||
except Exception as e:
|
|
||||||
print('Fusion Failed. ' + str(e))
|
|
||||||
MW.addMessage.emit('Fusion Failed. ' + str(e), 'error', True)
|
|
||||||
elif len(currentJobs) > 1 and options.title != 'defaulttitle':
|
|
||||||
currentJobs.clear()
|
|
||||||
error_message = 'Process Failed. Custom title can\'t be set when processing more than 1 source.\nDid you forget to check fusion?'
|
|
||||||
print(error_message)
|
|
||||||
MW.addMessage.emit(error_message, 'error', True)
|
|
||||||
for i, job in enumerate(currentJobs, start=1):
|
|
||||||
job_progress_number = f'[{i}/{len(currentJobs)}] '
|
|
||||||
sleep(0.5)
|
sleep(0.5)
|
||||||
if not self.conversionAlive:
|
if not self.conversionAlive:
|
||||||
self.clean()
|
self.clean()
|
||||||
return
|
return
|
||||||
self.errors = False
|
self.errors = False
|
||||||
MW.addMessage.emit(f'<b>{job_progress_number}Source:</b> ' + job, 'info', False)
|
MW.addMessage.emit('<b>Source:</b> ' + job, 'info', False)
|
||||||
if gui_current_format == 'CBZ':
|
if gui_current_format == 'CBZ':
|
||||||
MW.addMessage.emit('Creating CBZ files', 'info', False)
|
MW.addMessage.emit('Creating CBZ files', 'info', False)
|
||||||
GUI.progress.content = 'Creating CBZ files'
|
GUI.progress.content = 'Creating CBZ files'
|
||||||
elif gui_current_format == 'PDF':
|
|
||||||
MW.addMessage.emit('Creating PDF files', 'info', False)
|
|
||||||
GUI.progress.content = 'Creating PDF files'
|
|
||||||
else:
|
else:
|
||||||
MW.addMessage.emit('Creating EPUB files', 'info', False)
|
MW.addMessage.emit('Creating EPUB files', 'info', False)
|
||||||
GUI.progress.content = 'Creating EPUB files'
|
GUI.progress.content = 'Creating EPUB files'
|
||||||
@@ -409,7 +296,7 @@ class WorkerThread(QThread):
|
|||||||
jobargv.append(job)
|
jobargv.append(job)
|
||||||
try:
|
try:
|
||||||
comic2ebook.options = comic2ebook.checkOptions(copy(options))
|
comic2ebook.options = comic2ebook.checkOptions(copy(options))
|
||||||
outputPath = comic2ebook.makeBook(job, self, job_progress_number)
|
outputPath = comic2ebook.makeBook(job, self)
|
||||||
MW.hideProgressBar.emit()
|
MW.hideProgressBar.emit()
|
||||||
except UserWarning as warn:
|
except UserWarning as warn:
|
||||||
if not self.conversionAlive:
|
if not self.conversionAlive:
|
||||||
@@ -427,8 +314,13 @@ class WorkerThread(QThread):
|
|||||||
GUI.progress.content = ''
|
GUI.progress.content = ''
|
||||||
self.errors = True
|
self.errors = True
|
||||||
_, _, traceback = sys.exc_info()
|
_, _, traceback = sys.exc_info()
|
||||||
MW.showDialog.emit("Error during conversion %s:\n\n%s\n\nTraceback:\n%s"
|
if len(err.args) == 1:
|
||||||
% (jobargv[-1], str(err), sanitizeTrace(traceback)), 'error')
|
MW.showDialog.emit("Error during conversion %s:\n\n%s\n\nTraceback:\n%s"
|
||||||
|
% (jobargv[-1], str(err), sanitizeTrace(traceback)), 'error')
|
||||||
|
else:
|
||||||
|
MW.showDialog.emit("Error during conversion %s:\n\n%s\n\nTraceback:\n%s"
|
||||||
|
% (jobargv[-1], str(err.args[0]), err.args[1]), 'error')
|
||||||
|
GUI.sentry.extra_context({'realTraceback': err.args[1]})
|
||||||
if ' is corrupted.' not in str(err):
|
if ' is corrupted.' not in str(err):
|
||||||
GUI.sentry.captureException()
|
GUI.sentry.captureException()
|
||||||
MW.addMessage.emit('Error during conversion! Please consult '
|
MW.addMessage.emit('Error during conversion! Please consult '
|
||||||
@@ -446,12 +338,10 @@ class WorkerThread(QThread):
|
|||||||
GUI.progress.content = ''
|
GUI.progress.content = ''
|
||||||
if gui_current_format == 'CBZ':
|
if gui_current_format == 'CBZ':
|
||||||
MW.addMessage.emit('Creating CBZ files... <b>Done!</b>', 'info', True)
|
MW.addMessage.emit('Creating CBZ files... <b>Done!</b>', 'info', True)
|
||||||
elif gui_current_format == 'PDF':
|
|
||||||
MW.addMessage.emit('Creating PDF files... <b>Done!</b>', 'info', True)
|
|
||||||
else:
|
else:
|
||||||
MW.addMessage.emit('Creating EPUB files... <b>Done!</b>', 'info', True)
|
MW.addMessage.emit('Creating EPUB files... <b>Done!</b>', 'info', True)
|
||||||
if 'MOBI' in gui_current_format:
|
if 'MOBI' in gui_current_format:
|
||||||
MW.progressBarTick.emit(f'{job_progress_number}Creating MOBI files')
|
MW.progressBarTick.emit('Creating MOBI files')
|
||||||
MW.progressBarTick.emit(str(len(outputPath) * 2 + 1))
|
MW.progressBarTick.emit(str(len(outputPath) * 2 + 1))
|
||||||
MW.progressBarTick.emit('tick')
|
MW.progressBarTick.emit('tick')
|
||||||
MW.addMessage.emit('Creating MOBI files', 'info', False)
|
MW.addMessage.emit('Creating MOBI files', 'info', False)
|
||||||
@@ -501,10 +391,8 @@ class WorkerThread(QThread):
|
|||||||
k = kindle.Kindle(options.profile)
|
k = kindle.Kindle(options.profile)
|
||||||
if k.path and k.coverSupport:
|
if k.path and k.coverSupport:
|
||||||
for item in outputPath:
|
for item in outputPath:
|
||||||
cover = comic2ebook.options.covers[outputPath.index(item)][0]
|
comic2ebook.options.covers[outputPath.index(item)][0].saveToKindle(
|
||||||
if cover:
|
k, comic2ebook.options.covers[outputPath.index(item)][1])
|
||||||
cover.saveToKindle(
|
|
||||||
k, comic2ebook.options.covers[outputPath.index(item)][1])
|
|
||||||
MW.addMessage.emit('Kindle detected. Uploading covers... <b>Done!</b>', 'info', False)
|
MW.addMessage.emit('Kindle detected. Uploading covers... <b>Done!</b>', 'info', False)
|
||||||
else:
|
else:
|
||||||
GUI.progress.content = ''
|
GUI.progress.content = ''
|
||||||
@@ -525,16 +413,13 @@ class WorkerThread(QThread):
|
|||||||
if os.path.exists(item.replace('.epub', '.mobi')):
|
if os.path.exists(item.replace('.epub', '.mobi')):
|
||||||
os.remove(item.replace('.epub', '.mobi'))
|
os.remove(item.replace('.epub', '.mobi'))
|
||||||
MW.addMessage.emit('KindleGen failed to create MOBI!', 'error', False)
|
MW.addMessage.emit('KindleGen failed to create MOBI!', 'error', False)
|
||||||
MW.addMessage.emit(self.kindlegenErrorCode[1], 'error', False)
|
|
||||||
MW.addTrayMessage.emit('KindleGen failed to create MOBI!', 'Critical')
|
MW.addTrayMessage.emit('KindleGen failed to create MOBI!', 'Critical')
|
||||||
if self.kindlegenErrorCode[0] == 1 and self.kindlegenErrorCode[1] != '':
|
if self.kindlegenErrorCode[0] == 1 and self.kindlegenErrorCode[1] != '':
|
||||||
MW.showDialog.emit("KindleGen error:\n\n" + self.kindlegenErrorCode[1], 'error')
|
MW.showDialog.emit("KindleGen error:\n\n" + self.kindlegenErrorCode[1], 'error')
|
||||||
if self.kindlegenErrorCode[0] == 23026:
|
if self.kindlegenErrorCode[0] == 23026:
|
||||||
MW.addMessage.emit('Created EPUB file was too big. Weird file structure?', 'error', False)
|
MW.addMessage.emit('Created EPUB file was too big.', 'error', False)
|
||||||
MW.addMessage.emit('EPUB file: ' + str(epubSize) + 'MB. Supported size: ~350MB.', 'error',
|
MW.addMessage.emit('EPUB file: ' + str(epubSize) + 'MB. Supported size: ~350MB.', 'error',
|
||||||
False)
|
False)
|
||||||
if self.kindlegenErrorCode[0] == 3221226505:
|
|
||||||
MW.addMessage.emit('Unknown Windows error. Possibly filepath too long?', 'error', False)
|
|
||||||
else:
|
else:
|
||||||
for item in outputPath:
|
for item in outputPath:
|
||||||
if GUI.targetDirectory and GUI.targetDirectory != os.path.dirname(item):
|
if GUI.targetDirectory and GUI.targetDirectory != os.path.dirname(item):
|
||||||
@@ -542,12 +427,6 @@ class WorkerThread(QThread):
|
|||||||
move(item, GUI.targetDirectory)
|
move(item, GUI.targetDirectory)
|
||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
if options.filefusion:
|
|
||||||
for path in currentJobs:
|
|
||||||
if os.path.isfile(path):
|
|
||||||
os.remove(path)
|
|
||||||
elif os.path.isdir(path):
|
|
||||||
rmtree(path, True)
|
|
||||||
GUI.progress.content = ''
|
GUI.progress.content = ''
|
||||||
GUI.progress.stop()
|
GUI.progress.stop()
|
||||||
MW.hideProgressBar.emit()
|
MW.hideProgressBar.emit()
|
||||||
@@ -577,33 +456,17 @@ class SystemTrayIcon(QSystemTrayIcon):
|
|||||||
|
|
||||||
|
|
||||||
class KCCGUI(KCC_ui.Ui_mainWindow):
|
class KCCGUI(KCC_ui.Ui_mainWindow):
|
||||||
def selectDefaultOutputFolder(self):
|
def selectDir(self):
|
||||||
dname = QFileDialog.getExistingDirectory(MW, 'Select default output folder', self.defaultOutputFolder)
|
if self.needClean:
|
||||||
if self.is_directory_on_kindle(dname):
|
self.needClean = False
|
||||||
return
|
GUI.jobList.clear()
|
||||||
|
dname = QFileDialog.getExistingDirectory(MW, 'Select directory', self.lastPath)
|
||||||
if dname != '':
|
if dname != '':
|
||||||
if sys.platform.startswith('win'):
|
if sys.platform.startswith('win'):
|
||||||
dname = dname.replace('/', '\\')
|
dname = dname.replace('/', '\\')
|
||||||
GUI.defaultOutputFolder = dname
|
self.lastPath = os.path.abspath(os.path.join(dname, os.pardir))
|
||||||
|
GUI.jobList.addItem(dname)
|
||||||
def is_directory_on_kindle(self, dname):
|
GUI.jobList.scrollToBottom()
|
||||||
path = Path(dname)
|
|
||||||
for parent in itertools.chain([path], path.parents):
|
|
||||||
if parent.name == 'documents' and parent.parent.joinpath('system').joinpath('thumbnails').is_dir():
|
|
||||||
self.addMessage("Cannot select Kindle as output directory", 'error')
|
|
||||||
return True
|
|
||||||
|
|
||||||
def selectOutputFolder(self):
|
|
||||||
dname = QFileDialog.getExistingDirectory(MW, 'Select output directory', self.lastPath)
|
|
||||||
if self.is_directory_on_kindle(dname):
|
|
||||||
return
|
|
||||||
if dname != '':
|
|
||||||
if sys.platform.startswith('win'):
|
|
||||||
dname = dname.replace('/', '\\')
|
|
||||||
GUI.targetDirectory = dname
|
|
||||||
else:
|
|
||||||
GUI.targetDirectory = ''
|
|
||||||
return GUI.targetDirectory
|
|
||||||
|
|
||||||
def selectFile(self):
|
def selectFile(self):
|
||||||
if self.needClean:
|
if self.needClean:
|
||||||
@@ -617,50 +480,37 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
'Comic (*.pdf);;All (*.*)')
|
'Comic (*.pdf);;All (*.*)')
|
||||||
for fname in fnames[0]:
|
for fname in fnames[0]:
|
||||||
if fname != '':
|
if fname != '':
|
||||||
|
if sys.platform.startswith('win'):
|
||||||
|
fname = fname.replace('/', '\\')
|
||||||
self.lastPath = os.path.abspath(os.path.join(fname, os.pardir))
|
self.lastPath = os.path.abspath(os.path.join(fname, os.pardir))
|
||||||
GUI.jobList.addItem(fname)
|
GUI.jobList.addItem(fname)
|
||||||
GUI.jobList.scrollToBottom()
|
GUI.jobList.scrollToBottom()
|
||||||
|
|
||||||
def selectDir(self):
|
def selectFileMetaEditor(self):
|
||||||
if self.needClean:
|
sname = ''
|
||||||
self.needClean = False
|
if QApplication.keyboardModifiers() == Qt.ShiftModifier:
|
||||||
GUI.jobList.clear()
|
dname = QFileDialog.getExistingDirectory(MW, 'Select directory', self.lastPath)
|
||||||
|
if dname != '':
|
||||||
dialog = QFileDialog(MW, 'Select input folder(s)', self.lastPath)
|
sname = os.path.join(dname, 'ComicInfo.xml')
|
||||||
dialog.setFileMode(QFileDialog.FileMode.Directory)
|
if sys.platform.startswith('win'):
|
||||||
dialog.setOption(QFileDialog.Option.ShowDirsOnly, True)
|
sname = sname.replace('/', '\\')
|
||||||
dialog.setOption(QFileDialog.Option.DontUseNativeDialog, True)
|
self.lastPath = os.path.abspath(sname)
|
||||||
dialog.findChild(QTreeView).setSelectionMode(QAbstractItemView.ExtendedSelection)
|
else:
|
||||||
|
if self.sevenzip:
|
||||||
if dialog.exec():
|
fname = QFileDialog.getOpenFileName(MW, 'Select file', self.lastPath,
|
||||||
dnames = dialog.selectedFiles()
|
'Comic (*.cbz *.cbr *.cb7)')
|
||||||
for dname in dnames:
|
|
||||||
if dname != '':
|
|
||||||
self.lastPath = os.path.abspath(os.path.join(dname, os.pardir))
|
|
||||||
GUI.jobList.addItem(dname)
|
|
||||||
GUI.jobList.scrollToBottom()
|
|
||||||
|
|
||||||
|
|
||||||
def selectFileMetaEditor(self, sname):
|
|
||||||
if not sname:
|
|
||||||
if QApplication.keyboardModifiers() == Qt.ShiftModifier:
|
|
||||||
dname = QFileDialog.getExistingDirectory(MW, 'Select directory', self.lastPath)
|
|
||||||
if dname != '':
|
|
||||||
sname = os.path.join(dname, 'ComicInfo.xml')
|
|
||||||
self.lastPath = os.path.dirname(sname)
|
|
||||||
else:
|
else:
|
||||||
if self.sevenzip:
|
fname = ['']
|
||||||
fname = QFileDialog.getOpenFileName(MW, 'Select file', self.lastPath,
|
self.showDialog("Editor is disabled due to a lack of 7z.", 'error')
|
||||||
'Comic (*.cbz *.cbr *.cb7)')
|
self.addMessage('<a href="https://github.com/ciromattia/kcc#7-zip">Install 7z (link)</a>'
|
||||||
|
' to enable metadata editing.', 'warning')
|
||||||
|
if fname[0] != '':
|
||||||
|
if sys.platform.startswith('win'):
|
||||||
|
sname = fname[0].replace('/', '\\')
|
||||||
else:
|
else:
|
||||||
fname = ['']
|
|
||||||
self.showDialog("Editor is disabled due to a lack of 7z.", 'error')
|
|
||||||
self.addMessage('<a href="https://github.com/ciromattia/kcc#7-zip">Install 7z (link)</a>'
|
|
||||||
' to enable metadata editing.', 'warning')
|
|
||||||
if fname[0] != '':
|
|
||||||
sname = fname[0]
|
sname = fname[0]
|
||||||
self.lastPath = os.path.abspath(os.path.join(sname, os.pardir))
|
self.lastPath = os.path.abspath(os.path.join(sname, os.pardir))
|
||||||
if sname:
|
if sname != '':
|
||||||
try:
|
try:
|
||||||
self.editor.loadData(sname)
|
self.editor.loadData(sname)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
@@ -678,10 +528,6 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
# noinspection PyCallByClass
|
# noinspection PyCallByClass
|
||||||
QDesktopServices.openUrl(QUrl('https://github.com/ciromattia/kcc/wiki'))
|
QDesktopServices.openUrl(QUrl('https://github.com/ciromattia/kcc/wiki'))
|
||||||
|
|
||||||
def openKofi(self):
|
|
||||||
# noinspection PyCallByClass
|
|
||||||
QDesktopServices.openUrl(QUrl('https://ko-fi.com/eink_dude'))
|
|
||||||
|
|
||||||
def modeChange(self, mode):
|
def modeChange(self, mode):
|
||||||
if mode == 1:
|
if mode == 1:
|
||||||
self.currentMode = 1
|
self.currentMode = 1
|
||||||
@@ -704,7 +550,7 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
GUI.editorButton.setEnabled(status)
|
GUI.editorButton.setEnabled(status)
|
||||||
GUI.wikiButton.setEnabled(status)
|
GUI.wikiButton.setEnabled(status)
|
||||||
GUI.deviceBox.setEnabled(status)
|
GUI.deviceBox.setEnabled(status)
|
||||||
GUI.defaultOutputFolderButton.setEnabled(status)
|
GUI.directoryButton.setEnabled(status)
|
||||||
GUI.clearButton.setEnabled(status)
|
GUI.clearButton.setEnabled(status)
|
||||||
GUI.fileButton.setEnabled(status)
|
GUI.fileButton.setEnabled(status)
|
||||||
GUI.formatBox.setEnabled(status)
|
GUI.formatBox.setEnabled(status)
|
||||||
@@ -749,99 +595,36 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
GUI.croppingWidget.setVisible(False)
|
GUI.croppingWidget.setVisible(False)
|
||||||
self.changeCroppingPower(100) # 1.0
|
self.changeCroppingPower(100) # 1.0
|
||||||
|
|
||||||
def togglejpegqualityBox(self, value):
|
|
||||||
if value:
|
|
||||||
GUI.jpegQualityWidget.setVisible(True)
|
|
||||||
else:
|
|
||||||
GUI.jpegQualityWidget.setVisible(False)
|
|
||||||
|
|
||||||
def togglewebtoonBox(self, value):
|
def togglewebtoonBox(self, value):
|
||||||
if value:
|
if value:
|
||||||
self.addMessage('You can choose a taller device profile to get taller cuts in webtoon mode.', 'info')
|
|
||||||
self.addMessage('Try reading webtoon panels side by side in landscape!', 'info')
|
|
||||||
GUI.qualityBox.setEnabled(False)
|
GUI.qualityBox.setEnabled(False)
|
||||||
GUI.qualityBox.setChecked(False)
|
GUI.qualityBox.setChecked(False)
|
||||||
GUI.mangaBox.setEnabled(False)
|
GUI.mangaBox.setEnabled(False)
|
||||||
GUI.mangaBox.setChecked(False)
|
GUI.mangaBox.setChecked(False)
|
||||||
GUI.rotateBox.setEnabled(False)
|
GUI.rotateBox.setEnabled(False)
|
||||||
GUI.rotateBox.setChecked(False)
|
GUI.rotateBox.setChecked(False)
|
||||||
GUI.borderBox.setEnabled(False)
|
|
||||||
GUI.borderBox.setCheckState(Qt.CheckState.PartiallyChecked)
|
|
||||||
GUI.upscaleBox.setEnabled(False)
|
GUI.upscaleBox.setEnabled(False)
|
||||||
GUI.upscaleBox.setChecked(False)
|
GUI.upscaleBox.setChecked(True)
|
||||||
GUI.croppingBox.setEnabled(False)
|
|
||||||
GUI.croppingBox.setChecked(False)
|
|
||||||
GUI.interPanelCropBox.setEnabled(False)
|
|
||||||
GUI.interPanelCropBox.setChecked(False)
|
|
||||||
GUI.autoLevelBox.setEnabled(False)
|
|
||||||
GUI.autoLevelBox.setChecked(False)
|
|
||||||
GUI.autocontrastBox.setEnabled(False)
|
|
||||||
GUI.autocontrastBox.setChecked(False)
|
|
||||||
else:
|
else:
|
||||||
profile = GUI.profiles[str(GUI.deviceBox.currentText())]
|
profile = GUI.profiles[str(GUI.deviceBox.currentText())]
|
||||||
if profile['PVOptions']:
|
if profile['PVOptions']:
|
||||||
GUI.qualityBox.setEnabled(True)
|
GUI.qualityBox.setEnabled(True)
|
||||||
GUI.mangaBox.setEnabled(True)
|
GUI.mangaBox.setEnabled(True)
|
||||||
GUI.rotateBox.setEnabled(True)
|
GUI.rotateBox.setEnabled(True)
|
||||||
GUI.borderBox.setEnabled(True)
|
GUI.upscaleBox.setEnabled(True)
|
||||||
profile = GUI.profiles[str(GUI.deviceBox.currentText())]
|
|
||||||
if not profile['Label'].startswith('KS'):
|
|
||||||
GUI.upscaleBox.setEnabled(True)
|
|
||||||
GUI.croppingBox.setEnabled(True)
|
|
||||||
GUI.interPanelCropBox.setEnabled(True)
|
|
||||||
GUI.autoLevelBox.setEnabled(True)
|
|
||||||
GUI.autocontrastBox.setEnabled(True)
|
|
||||||
GUI.autocontrastBox.setChecked(True)
|
|
||||||
|
|
||||||
|
|
||||||
def togglequalityBox(self, value):
|
def togglequalityBox(self, value):
|
||||||
profile = GUI.profiles[str(GUI.deviceBox.currentText())]
|
profile = GUI.profiles[str(GUI.deviceBox.currentText())]
|
||||||
if value == 2:
|
if value == 2:
|
||||||
if profile['Label'] not in ('K57', 'KPW', 'K810') :
|
if profile['Label'] in ['KV', 'KO']:
|
||||||
self.addMessage('This option is intended for older Kindle models.', 'warning')
|
self.addMessage('This option is intended for older Kindle models.', 'warning')
|
||||||
self.addMessage('On this device, there will be conversion speed and quality issues.', 'warning')
|
self.addMessage('On this device, quality improvement will be negligible.', 'warning')
|
||||||
self.addMessage('Use the Kindle Scribe profile if you want higher resolution when zooming.', 'warning')
|
|
||||||
GUI.upscaleBox.setEnabled(False)
|
GUI.upscaleBox.setEnabled(False)
|
||||||
GUI.upscaleBox.setChecked(True)
|
GUI.upscaleBox.setChecked(True)
|
||||||
else:
|
else:
|
||||||
GUI.upscaleBox.setEnabled(True)
|
GUI.upscaleBox.setEnabled(True)
|
||||||
GUI.upscaleBox.setChecked(profile['DefaultUpscale'])
|
GUI.upscaleBox.setChecked(profile['DefaultUpscale'])
|
||||||
|
|
||||||
def toggleImageFormatBox(self, value):
|
|
||||||
profile = GUI.profiles[str(GUI.deviceBox.currentText())]
|
|
||||||
if value == 1:
|
|
||||||
if profile['Label'].startswith('KS'):
|
|
||||||
current_format = GUI.formats[str(GUI.formatBox.currentText())]['format']
|
|
||||||
for bad_format in ('MOBI', 'EPUB'):
|
|
||||||
if bad_format in current_format:
|
|
||||||
self.addMessage('Scribe PNG MOBI/EPUB has a lot of problems like blank pages/sections. Use JPG instead.', 'warning')
|
|
||||||
break
|
|
||||||
|
|
||||||
def togglechunkSizeCheckBox(self, value):
|
|
||||||
GUI.chunkSizeWidget.setVisible(value)
|
|
||||||
|
|
||||||
def toggletitleEdit(self, value):
|
|
||||||
if value:
|
|
||||||
self.metadataTitleBox.setChecked(False)
|
|
||||||
|
|
||||||
def togglefileFusionBox(self, value):
|
|
||||||
if value:
|
|
||||||
GUI.metadataTitleBox.setChecked(False)
|
|
||||||
GUI.metadataTitleBox.setEnabled(False)
|
|
||||||
else:
|
|
||||||
GUI.metadataTitleBox.setEnabled(True)
|
|
||||||
|
|
||||||
def togglemetadataTitleBox(self, value):
|
|
||||||
if value:
|
|
||||||
GUI.titleEdit.setText(None)
|
|
||||||
|
|
||||||
def editSourceMetadata(self, item):
|
|
||||||
if item.icon().isNull():
|
|
||||||
sname = item.text()
|
|
||||||
if os.path.isdir(sname):
|
|
||||||
sname = os.path.join(sname, "ComicInfo.xml")
|
|
||||||
self.selectFileMetaEditor(sname)
|
|
||||||
|
|
||||||
def changeGamma(self, value):
|
def changeGamma(self, value):
|
||||||
valueRaw = int(5 * round(float(value) / 5))
|
valueRaw = int(5 * round(float(value) / 5))
|
||||||
value = '%.2f' % (float(valueRaw) / 100)
|
value = '%.2f' % (float(valueRaw) / 100)
|
||||||
@@ -869,20 +652,12 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
self.modeChange(1)
|
self.modeChange(1)
|
||||||
GUI.colorBox.setChecked(profile['ForceColor'])
|
GUI.colorBox.setChecked(profile['ForceColor'])
|
||||||
self.changeFormat()
|
self.changeFormat()
|
||||||
|
GUI.gammaSlider.setValue(0)
|
||||||
|
self.changeGamma(0)
|
||||||
if not GUI.webtoonBox.isChecked():
|
if not GUI.webtoonBox.isChecked():
|
||||||
GUI.qualityBox.setEnabled(profile['PVOptions'])
|
GUI.qualityBox.setEnabled(profile['PVOptions'])
|
||||||
GUI.upscaleBox.setChecked(profile['DefaultUpscale'])
|
GUI.upscaleBox.setChecked(profile['DefaultUpscale'])
|
||||||
if profile['Label'].startswith('KS'):
|
GUI.mangaBox.setChecked(True)
|
||||||
GUI.upscaleBox.setDisabled(True)
|
|
||||||
else:
|
|
||||||
if not GUI.webtoonBox.isChecked():
|
|
||||||
GUI.upscaleBox.setEnabled(True)
|
|
||||||
if profile['Label'] == 'KCS':
|
|
||||||
current_format = GUI.formats[str(GUI.formatBox.currentText())]['format']
|
|
||||||
for bad_format in ('MOBI', 'EPUB'):
|
|
||||||
if bad_format in current_format:
|
|
||||||
self.addMessage('Colorsoft MOBI/EPUB can have blank pages. Just go back a few pages, exit, and reenter book.', 'info')
|
|
||||||
break
|
|
||||||
if not profile['PVOptions']:
|
if not profile['PVOptions']:
|
||||||
GUI.qualityBox.setChecked(False)
|
GUI.qualityBox.setChecked(False)
|
||||||
if str(GUI.deviceBox.currentText()) == 'Other':
|
if str(GUI.deviceBox.currentText()) == 'Other':
|
||||||
@@ -902,14 +677,6 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
else:
|
else:
|
||||||
GUI.outputSplit.setEnabled(False)
|
GUI.outputSplit.setEnabled(False)
|
||||||
GUI.outputSplit.setChecked(False)
|
GUI.outputSplit.setChecked(False)
|
||||||
if (GUI.formats[str(GUI.formatBox.currentText())]['format'] == 'EPUB-200MB' or
|
|
||||||
GUI.formats[str(GUI.formatBox.currentText())]['format'] == 'MOBI+EPUB-200MB'):
|
|
||||||
GUI.chunkSizeCheckBox.setEnabled(False)
|
|
||||||
GUI.chunkSizeCheckBox.setChecked(False)
|
|
||||||
elif not GUI.webtoonBox.isChecked():
|
|
||||||
GUI.chunkSizeCheckBox.setEnabled(True)
|
|
||||||
if GUI.formats[str(GUI.formatBox.currentText())]['format'] in ('CBZ', 'PDF') and not GUI.webtoonBox.isChecked():
|
|
||||||
self.addMessage("Partially check W/B Margins if you don't want KCC to extend the image margins.", 'info')
|
|
||||||
|
|
||||||
def stripTags(self, html):
|
def stripTags(self, html):
|
||||||
s = HTMLStripper()
|
s = HTMLStripper()
|
||||||
@@ -964,10 +731,13 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
self.worker.sync()
|
self.worker.sync()
|
||||||
else:
|
else:
|
||||||
if QApplication.keyboardModifiers() == Qt.KeyboardModifier.ShiftModifier:
|
if QApplication.keyboardModifiers() == Qt.KeyboardModifier.ShiftModifier:
|
||||||
if not self.selectOutputFolder():
|
dname = QFileDialog.getExistingDirectory(MW, 'Select output directory', self.lastPath)
|
||||||
return
|
if dname != '':
|
||||||
elif GUI.defaultOutputFolderBox.isChecked():
|
if sys.platform.startswith('win'):
|
||||||
self.targetDirectory = self.defaultOutputFolder
|
dname = dname.replace('/', '\\')
|
||||||
|
GUI.targetDirectory = dname
|
||||||
|
else:
|
||||||
|
GUI.targetDirectory = ''
|
||||||
else:
|
else:
|
||||||
GUI.targetDirectory = ''
|
GUI.targetDirectory = ''
|
||||||
self.progress.start()
|
self.progress.start()
|
||||||
@@ -978,12 +748,6 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
self.addMessage('No files selected! Please choose files to convert.', 'error')
|
self.addMessage('No files selected! Please choose files to convert.', 'error')
|
||||||
self.needClean = True
|
self.needClean = True
|
||||||
return
|
return
|
||||||
if GUI.defaultOutputFolderBox.checkState() == Qt.CheckState.PartiallyChecked:
|
|
||||||
parent = Path(self.jobList.item(0).text()).parent
|
|
||||||
target_path = parent.joinpath(f"{parent.name}")
|
|
||||||
if not target_path.exists():
|
|
||||||
target_path.mkdir()
|
|
||||||
self.targetDirectory = str(target_path)
|
|
||||||
if self.currentMode > 2 and (GUI.widthBox.value() == 0 or GUI.heightBox.value() == 0):
|
if self.currentMode > 2 and (GUI.widthBox.value() == 0 or GUI.heightBox.value() == 0):
|
||||||
GUI.jobList.clear()
|
GUI.jobList.clear()
|
||||||
self.addMessage('Target resolution is not set!', 'error')
|
self.addMessage('Target resolution is not set!', 'error')
|
||||||
@@ -1015,46 +779,30 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
event.ignore()
|
event.ignore()
|
||||||
self.settings.setValue('settingsVersion', __version__)
|
self.settings.setValue('settingsVersion', __version__)
|
||||||
self.settings.setValue('lastPath', self.lastPath)
|
self.settings.setValue('lastPath', self.lastPath)
|
||||||
self.settings.setValue('defaultOutputFolder', self.defaultOutputFolder)
|
|
||||||
self.settings.setValue('lastDevice', GUI.deviceBox.currentIndex())
|
self.settings.setValue('lastDevice', GUI.deviceBox.currentIndex())
|
||||||
self.settings.setValue('currentFormat', GUI.formatBox.currentIndex())
|
self.settings.setValue('currentFormat', GUI.formatBox.currentIndex())
|
||||||
self.settings.setValue('startNumber', self.startNumber + 1)
|
self.settings.setValue('startNumber', self.startNumber + 1)
|
||||||
self.settings.setValue('windowSize', str(MW.size().width()) + 'x' + str(MW.size().height()))
|
self.settings.setValue('windowSize', str(MW.size().width()) + 'x' + str(MW.size().height()))
|
||||||
self.settings.setValue('options', {'mangaBox': GUI.mangaBox.checkState(),
|
self.settings.setValue('options', {'mangaBox': GUI.mangaBox.checkState().value,
|
||||||
'rotateBox': GUI.rotateBox.checkState(),
|
'rotateBox': GUI.rotateBox.checkState().value,
|
||||||
'qualityBox': GUI.qualityBox.checkState(),
|
'qualityBox': GUI.qualityBox.checkState().value,
|
||||||
'gammaBox': GUI.gammaBox.checkState(),
|
'gammaBox': GUI.gammaBox.checkState().value,
|
||||||
'autoLevelBox': GUI.autoLevelBox.checkState(),
|
'croppingBox': GUI.croppingBox.checkState().value,
|
||||||
'autocontrastBox': GUI.autocontrastBox.checkState(),
|
|
||||||
'croppingBox': GUI.croppingBox.checkState(),
|
|
||||||
'croppingPowerSlider': float(self.croppingPowerValue) * 100,
|
'croppingPowerSlider': float(self.croppingPowerValue) * 100,
|
||||||
'preserveMarginBox': self.preserveMarginBox.value(),
|
'upscaleBox': GUI.upscaleBox.checkState().value,
|
||||||
'interPanelCropBox': GUI.interPanelCropBox.checkState(),
|
'borderBox': GUI.borderBox.checkState().value,
|
||||||
'upscaleBox': GUI.upscaleBox.checkState(),
|
'webtoonBox': GUI.webtoonBox.checkState().value,
|
||||||
'borderBox': GUI.borderBox.checkState(),
|
'outputSplit': GUI.outputSplit.checkState().value,
|
||||||
'webtoonBox': GUI.webtoonBox.checkState(),
|
'colorBox': GUI.colorBox.checkState().value,
|
||||||
'outputSplit': GUI.outputSplit.checkState(),
|
'disableProcessingBox': GUI.disableProcessingBox.checkState().value,
|
||||||
'colorBox': GUI.colorBox.checkState(),
|
'mozJpegBox': GUI.mozJpegBox.checkState().value,
|
||||||
'eraseRainbowBox': GUI.eraseRainbowBox.checkState(),
|
|
||||||
'disableProcessingBox': GUI.disableProcessingBox.checkState(),
|
|
||||||
'pdfExtractBox': GUI.pdfExtractBox.checkState(),
|
|
||||||
'coverFillBox': GUI.coverFillBox.checkState(),
|
|
||||||
'metadataTitleBox': GUI.metadataTitleBox.checkState(),
|
|
||||||
'mozJpegBox': GUI.mozJpegBox.checkState(),
|
|
||||||
'jpegQualityBox': GUI.jpegQualityBox.checkState(),
|
|
||||||
'jpegQuality': GUI.jpegQualitySpinBox.value(),
|
|
||||||
'widthBox': GUI.widthBox.value(),
|
'widthBox': GUI.widthBox.value(),
|
||||||
'heightBox': GUI.heightBox.value(),
|
'heightBox': GUI.heightBox.value(),
|
||||||
'deleteBox': GUI.deleteBox.checkState(),
|
'deleteBox': GUI.deleteBox.checkState().value,
|
||||||
'spreadShiftBox': GUI.spreadShiftBox.checkState(),
|
'spreadShiftBox': GUI.spreadShiftBox.checkState().value,
|
||||||
'fileFusionBox': GUI.fileFusionBox.checkState(),
|
'noRotateBox': GUI.noRotateBox.checkState().value,
|
||||||
'defaultOutputFolderBox': GUI.defaultOutputFolderBox.checkState(),
|
'maximizeStrips': GUI.maximizeStrips.checkState().value,
|
||||||
'noRotateBox': GUI.noRotateBox.checkState(),
|
'gammaSlider': float(self.gammaValue) * 100})
|
||||||
'rotateFirstBox': GUI.rotateFirstBox.checkState(),
|
|
||||||
'maximizeStrips': GUI.maximizeStrips.checkState(),
|
|
||||||
'gammaSlider': float(self.gammaValue) * 100,
|
|
||||||
'chunkSizeCheckBox': GUI.chunkSizeCheckBox.checkState(),
|
|
||||||
'chunkSizeBox': GUI.chunkSizeBox.value()})
|
|
||||||
self.settings.sync()
|
self.settings.sync()
|
||||||
self.tray.hide()
|
self.tray.hide()
|
||||||
|
|
||||||
@@ -1063,7 +811,7 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
MW.activateWindow()
|
MW.activateWindow()
|
||||||
if type(message) is bytes:
|
if type(message) is bytes:
|
||||||
message = message.decode('UTF-8')
|
message = message.decode('UTF-8')
|
||||||
if not self.conversionAlive and message != 'ARISE' and not GUI.jobList.findItems(message, Qt.MatchFlag.MatchExactly):
|
if not self.conversionAlive and message != 'ARISE':
|
||||||
if self.needClean:
|
if self.needClean:
|
||||||
self.needClean = False
|
self.needClean = False
|
||||||
GUI.jobList.clear()
|
GUI.jobList.clear()
|
||||||
@@ -1094,8 +842,6 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
if message[-1] == '/':
|
if message[-1] == '/':
|
||||||
message = message[:-1]
|
message = message[:-1]
|
||||||
self.handleMessage(message)
|
self.handleMessage(message)
|
||||||
# sorting may conflict with manual file fusion order
|
|
||||||
# GUI.jobList.sortItems()
|
|
||||||
|
|
||||||
def forceShutdown(self):
|
def forceShutdown(self):
|
||||||
self.saveSettings(None)
|
self.saveSettings(None)
|
||||||
@@ -1108,7 +854,7 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
try:
|
try:
|
||||||
versionCheck = subprocess_run(['kindlegen', '-locale', 'en'], stdout=PIPE, stderr=STDOUT, encoding='UTF-8', errors='ignore', check=True)
|
versionCheck = subprocess_run(['kindlegen', '-locale', 'en'], stdout=PIPE, stderr=STDOUT, encoding='UTF-8')
|
||||||
self.kindleGen = True
|
self.kindleGen = True
|
||||||
for line in versionCheck.stdout.splitlines():
|
for line in versionCheck.stdout.splitlines():
|
||||||
if 'Amazon kindlegen' in line:
|
if 'Amazon kindlegen' in line:
|
||||||
@@ -1117,7 +863,7 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
self.addMessage('Your <a href="https://www.amazon.com/b?node=23496309011">KindleGen</a>'
|
self.addMessage('Your <a href="https://www.amazon.com/b?node=23496309011">KindleGen</a>'
|
||||||
' is outdated! MOBI conversion might fail.', 'warning')
|
' is outdated! MOBI conversion might fail.', 'warning')
|
||||||
break
|
break
|
||||||
except (FileNotFoundError, CalledProcessError):
|
except FileNotFoundError:
|
||||||
self.kindleGen = False
|
self.kindleGen = False
|
||||||
if startup:
|
if startup:
|
||||||
self.display_kindlegen_missing()
|
self.display_kindlegen_missing()
|
||||||
@@ -1130,21 +876,14 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
self.setupUi(MW)
|
self.setupUi(MW)
|
||||||
self.editor = KCCGUI_MetaEditor()
|
self.editor = KCCGUI_MetaEditor()
|
||||||
self.icons = Icons()
|
self.icons = Icons()
|
||||||
self.settings = QSettings('ciromattia', 'kcc9')
|
self.settings = QSettings('ciromattia', 'kcc')
|
||||||
self.settingsVersion = self.settings.value('settingsVersion', '', type=str)
|
self.settingsVersion = self.settings.value('settingsVersion', '', type=str)
|
||||||
self.lastPath = self.settings.value('lastPath', '', type=str)
|
self.lastPath = self.settings.value('lastPath', '', type=str)
|
||||||
self.defaultOutputFolder = str(self.settings.value('defaultOutputFolder', '', type=str))
|
|
||||||
if not os.path.exists(self.defaultOutputFolder):
|
|
||||||
self.defaultOutputFolder = ''
|
|
||||||
self.lastDevice = self.settings.value('lastDevice', 0, type=int)
|
self.lastDevice = self.settings.value('lastDevice', 0, type=int)
|
||||||
self.currentFormat = self.settings.value('currentFormat', 0, type=int)
|
self.currentFormat = self.settings.value('currentFormat', 0, type=int)
|
||||||
self.startNumber = self.settings.value('startNumber', 0, type=int)
|
self.startNumber = self.settings.value('startNumber', 0, type=int)
|
||||||
self.windowSize = self.settings.value('windowSize', '0x0', type=str)
|
self.windowSize = self.settings.value('windowSize', '0x0', type=str)
|
||||||
default_options = {'gammaSlider': 0, 'croppingBox': 2, 'croppingPowerSlider': 100}
|
self.options = self.settings.value('options', {'gammaSlider': 0, 'croppingBox': 2, 'croppingPowerSlider': 100})
|
||||||
try:
|
|
||||||
self.options = self.settings.value('options', default_options)
|
|
||||||
except Exception:
|
|
||||||
self.options = default_options
|
|
||||||
self.worker = WorkerThread()
|
self.worker = WorkerThread()
|
||||||
self.versionCheck = VersionThread()
|
self.versionCheck = VersionThread()
|
||||||
self.progress = ProgressThread()
|
self.progress = ProgressThread()
|
||||||
@@ -1168,7 +907,7 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
if self.windowSize == '0x0':
|
if self.windowSize == '0x0':
|
||||||
MW.resize(500, 500)
|
MW.resize(500, 500)
|
||||||
elif sys.platform.startswith('darwin'):
|
elif sys.platform.startswith('darwin'):
|
||||||
for element in ['editorButton', 'wikiButton', 'defaultOutputFolderButton', 'clearButton', 'fileButton', 'deviceBox',
|
for element in ['editorButton', 'wikiButton', 'directoryButton', 'clearButton', 'fileButton', 'deviceBox',
|
||||||
'convertButton', 'formatBox']:
|
'convertButton', 'formatBox']:
|
||||||
getattr(GUI, element).setMinimumSize(QSize(0, 0))
|
getattr(GUI, element).setMinimumSize(QSize(0, 0))
|
||||||
GUI.gridLayout.setContentsMargins(-1, -1, -1, -1)
|
GUI.gridLayout.setContentsMargins(-1, -1, -1, -1)
|
||||||
@@ -1181,59 +920,40 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
"MOBI/AZW3": {'icon': 'MOBI', 'format': 'MOBI'},
|
"MOBI/AZW3": {'icon': 'MOBI', 'format': 'MOBI'},
|
||||||
"EPUB": {'icon': 'EPUB', 'format': 'EPUB'},
|
"EPUB": {'icon': 'EPUB', 'format': 'EPUB'},
|
||||||
"CBZ": {'icon': 'CBZ', 'format': 'CBZ'},
|
"CBZ": {'icon': 'CBZ', 'format': 'CBZ'},
|
||||||
"PDF": {'icon': 'EPUB', 'format': 'PDF'},
|
"EPUB (Calibre KFX)": {'icon': 'EPUB', 'format': 'KFX'},
|
||||||
"KFX (does not work)": {'icon': 'KFX', 'format': 'KFX'},
|
|
||||||
"MOBI + EPUB": {'icon': 'MOBI', 'format': 'MOBI+EPUB'},
|
"MOBI + EPUB": {'icon': 'MOBI', 'format': 'MOBI+EPUB'},
|
||||||
"EPUB (200MB limit)": {'icon': 'EPUB', 'format': 'EPUB-200MB'},
|
"EPUB (200MB limit)": {'icon': 'EPUB', 'format': 'EPUB-200MB'}
|
||||||
"MOBI + EPUB (200MB limit)": {'icon': 'MOBI', 'format': 'MOBI+EPUB-200MB'},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
self.profiles = {
|
self.profiles = {
|
||||||
"Kindle Oasis 9/10": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
"Kindle Oasis 9/10": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
||||||
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KO'},
|
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KO'},
|
||||||
"Kindle 8/10": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
|
||||||
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'K810'},
|
|
||||||
"Kindle Oasis 8": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
"Kindle Oasis 8": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
||||||
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KPW34'},
|
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KV'},
|
||||||
"Kindle Voyage": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
"Kindle Voyage": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
||||||
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KV'},
|
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KV'},
|
||||||
"Kindle 1860x1920": {
|
"Kindle Scribe": {
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KS1860',
|
|
||||||
},
|
|
||||||
"Kindle 1920x1920": {
|
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KS1920',
|
|
||||||
},
|
|
||||||
"Kindle 1240x1860": {
|
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KS1240',
|
|
||||||
},
|
|
||||||
"Kindle Scribe 1/2": {
|
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KS',
|
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KS',
|
||||||
},
|
},
|
||||||
"Kindle Scribe 3": {
|
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KS3',
|
|
||||||
},
|
|
||||||
"Kindle Scribe Colorsoft": {
|
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': False, 'ForceColor': True, 'Label': 'KSCS',
|
|
||||||
},
|
|
||||||
"Kindle 11": {
|
"Kindle 11": {
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': False, 'Label': 'K11',
|
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': False, 'Label': 'K11',
|
||||||
},
|
},
|
||||||
"Kindle Paperwhite 11": {
|
"Kindle PW 11": {
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KPW5',
|
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KPW5',
|
||||||
},
|
},
|
||||||
"Kindle Paperwhite 12": {
|
"Kindle PW 12": {
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KO',
|
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KO',
|
||||||
},
|
},
|
||||||
"Kindle Colorsoft": {
|
"Kindle CS 12": {
|
||||||
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': True, 'Label': 'KCS',
|
'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0, 'DefaultUpscale': True, 'ForceColor': True, 'Label': 'KO',
|
||||||
},
|
},
|
||||||
"Kindle Paperwhite 7/10": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
"Kindle PW 7/10": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
||||||
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KPW34'},
|
'DefaultUpscale': True, 'ForceColor': False, 'Label': 'KV'},
|
||||||
"Kindle Paperwhite 5/6": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
"Kindle PW 5/6": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
||||||
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KPW'},
|
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KPW'},
|
||||||
"Kindle 4/5/7": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
"Kindle 4/5/7/8/10": {'PVOptions': True, 'ForceExpert': False, 'DefaultFormat': 0,
|
||||||
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'K57'},
|
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'K578'},
|
||||||
"Kindle DX": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 2,
|
"Kindle DX": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 2,
|
||||||
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KDX'},
|
'DefaultUpscale': False, 'ForceColor': False, 'Label': 'KDX'},
|
||||||
"Kobo Mini/Touch": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1,
|
"Kobo Mini/Touch": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1,
|
||||||
@@ -1278,24 +998,20 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
'Label': 'KoS'},
|
'Label': 'KoS'},
|
||||||
"Kobo Elipsa": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1, 'DefaultUpscale': True, 'ForceColor': False,
|
"Kobo Elipsa": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1, 'DefaultUpscale': True, 'ForceColor': False,
|
||||||
'Label': 'KoE'},
|
'Label': 'KoE'},
|
||||||
"reMarkable 1": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 3, 'DefaultUpscale': True, 'ForceColor': False,
|
"reMarkable 1": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1, 'DefaultUpscale': True, 'ForceColor': False,
|
||||||
'Label': 'Rmk1'},
|
'Label': 'Rmk1'},
|
||||||
"reMarkable 2": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 3, 'DefaultUpscale': True, 'ForceColor': False,
|
"reMarkable 2": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1, 'DefaultUpscale': True, 'ForceColor': False,
|
||||||
'Label': 'Rmk2'},
|
'Label': 'Rmk2'},
|
||||||
"reMarkable Paper Pro": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 3, 'DefaultUpscale': True, 'ForceColor': True,
|
"reMarkable Paper Pro": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 1, 'DefaultUpscale': True, 'ForceColor': True,
|
||||||
'Label': 'RmkPP'},
|
'Label': 'RmkPP'},
|
||||||
"reMarkable Paper Pro Move": {'PVOptions': False, 'ForceExpert': False, 'DefaultFormat': 3, 'DefaultUpscale': True, 'ForceColor': True,
|
|
||||||
'Label': 'RmkPPMove'},
|
|
||||||
"Other": {'PVOptions': False, 'ForceExpert': True, 'DefaultFormat': 1, 'DefaultUpscale': False, 'ForceColor': False,
|
"Other": {'PVOptions': False, 'ForceExpert': True, 'DefaultFormat': 1, 'DefaultUpscale': False, 'ForceColor': False,
|
||||||
'Label': 'OTHER'},
|
'Label': 'OTHER'},
|
||||||
}
|
}
|
||||||
profilesGUI = [
|
profilesGUI = [
|
||||||
"Kindle Scribe Colorsoft",
|
"Kindle CS 12",
|
||||||
"Kindle Scribe 3",
|
"Kindle PW 12",
|
||||||
"Kindle Colorsoft",
|
"Kindle Scribe",
|
||||||
"Kindle Paperwhite 12",
|
"Kindle PW 11",
|
||||||
"Kindle Scribe 1/2",
|
|
||||||
"Kindle Paperwhite 11",
|
|
||||||
"Kindle 11",
|
"Kindle 11",
|
||||||
"Kindle Oasis 9/10",
|
"Kindle Oasis 9/10",
|
||||||
"Separator",
|
"Separator",
|
||||||
@@ -1310,19 +1026,14 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
"reMarkable 1",
|
"reMarkable 1",
|
||||||
"reMarkable 2",
|
"reMarkable 2",
|
||||||
"reMarkable Paper Pro",
|
"reMarkable Paper Pro",
|
||||||
"reMarkable Paper Pro Move",
|
|
||||||
"Separator",
|
"Separator",
|
||||||
"Other",
|
"Other",
|
||||||
"Separator",
|
"Separator",
|
||||||
"Kindle 1920x1920",
|
|
||||||
"Kindle 1860x1920",
|
|
||||||
"Kindle 1240x1860",
|
|
||||||
"Kindle 8/10",
|
|
||||||
"Kindle Oasis 8",
|
"Kindle Oasis 8",
|
||||||
"Kindle Paperwhite 7/10",
|
"Kindle PW 7/10",
|
||||||
"Kindle Voyage",
|
"Kindle Voyage",
|
||||||
"Kindle Paperwhite 5/6",
|
"Kindle PW 5/6",
|
||||||
"Kindle 4/5/7",
|
"Kindle 4/5/7/8/10",
|
||||||
"Kindle Touch",
|
"Kindle Touch",
|
||||||
"Kindle Keyboard",
|
"Kindle Keyboard",
|
||||||
"Kindle DX",
|
"Kindle DX",
|
||||||
@@ -1341,60 +1052,50 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
"Kobo Mini/Touch",
|
"Kobo Mini/Touch",
|
||||||
]
|
]
|
||||||
|
|
||||||
link_dict = {
|
statusBarLabel = QLabel('<b><a href="https://kcc.iosphe.re/">HOMEPAGE</a> - <a href="https://github.'
|
||||||
'README': "https://github.com/ciromattia/kcc?tab=readme-ov-file#kcc",
|
'com/ciromattia/kcc/blob/master/README.md#issues--new-features--donations">DO'
|
||||||
'FAQ': "https://github.com/ciromattia/kcc/blob/master/README.md#faq",
|
'NATE</a> - <a href="http://www.mobileread.com/forums/showthread.php?t=207461'
|
||||||
'YOUTUBE': "https://youtu.be/IR2Fhcm9658?si=Z-2zzLaUFjmaEbrj",
|
'">FORUM</a></b>')
|
||||||
'COMMISSIONS': "https://github.com/ciromattia/kcc?tab=readme-ov-file#commissions",
|
|
||||||
'DONATE': "https://github.com/ciromattia/kcc/blob/master/README.md#issues--new-features--donations",
|
|
||||||
'FORUM': "http://www.mobileread.com/forums/showthread.php?t=207461",
|
|
||||||
'DISCORD': "https://discord.com/invite/qj7wpnUHav",
|
|
||||||
}
|
|
||||||
|
|
||||||
link_html_list = [f'<a href="{v}">{k}</a>' for k, v in link_dict.items()]
|
|
||||||
statusBarLabel = QLabel(f'<b>{" - ".join(link_html_list)}</b>')
|
|
||||||
statusBarLabel.setAlignment(Qt.AlignmentFlag.AlignCenter)
|
statusBarLabel.setAlignment(Qt.AlignmentFlag.AlignCenter)
|
||||||
statusBarLabel.setOpenExternalLinks(True)
|
statusBarLabel.setOpenExternalLinks(True)
|
||||||
GUI.statusBar.addPermanentWidget(statusBarLabel, 1)
|
GUI.statusBar.addPermanentWidget(statusBarLabel, 1)
|
||||||
|
|
||||||
self.addMessage('<b>Tip:</b> Hover mouse over options to see additional information in tooltips.', 'info')
|
self.addMessage('<b>Welcome!</b>', 'info')
|
||||||
self.addMessage('<b>Tip:</b> You can drag and drop image folders or comic files/archives into this window to convert.', 'info')
|
self.addMessage('<b>Remember:</b> All options have additional information in tooltips.', 'info')
|
||||||
if self.startNumber < 5:
|
if self.startNumber < 5:
|
||||||
self.addMessage('Since you are a new user of <b>KCC</b> please see few '
|
self.addMessage('Since you are a new user of <b>KCC</b> please see few '
|
||||||
'<a href="https://github.com/ciromattia/kcc/wiki/Important-tips">important tips</a>.',
|
'<a href="https://github.com/ciromattia/kcc/wiki/Important-tips">important tips</a>.',
|
||||||
'info')
|
'info')
|
||||||
|
try:
|
||||||
self.tar = TAR in available_archive_tools()
|
subprocess_run(['tar'], stdout=PIPE, stderr=STDOUT)
|
||||||
self.sevenzip = SEVENZIP in available_archive_tools()
|
self.tar = True
|
||||||
if not any([self.tar, self.sevenzip]):
|
except FileNotFoundError:
|
||||||
self.addMessage('<a href="https://github.com/ciromattia/kcc#7-zip">Install 7z (link)</a>'
|
self.tar = False
|
||||||
' to enable CBZ/CBR/ZIP/etc processing.', 'warning')
|
try:
|
||||||
|
subprocess_run(['7z'], stdout=PIPE, stderr=STDOUT)
|
||||||
|
self.sevenzip = True
|
||||||
|
except FileNotFoundError:
|
||||||
|
self.sevenzip = False
|
||||||
|
if not self.tar:
|
||||||
|
self.addMessage('<a href="https://github.com/ciromattia/kcc#7-zip">Install 7z (link)</a>'
|
||||||
|
' to enable CBZ/CBR/ZIP/etc processing.', 'warning')
|
||||||
self.detectKindleGen(True)
|
self.detectKindleGen(True)
|
||||||
|
|
||||||
APP.messageFromOtherInstance.connect(self.handleMessage)
|
APP.messageFromOtherInstance.connect(self.handleMessage)
|
||||||
GUI.defaultOutputFolderButton.clicked.connect(self.selectDefaultOutputFolder)
|
GUI.directoryButton.clicked.connect(self.selectDir)
|
||||||
GUI.clearButton.clicked.connect(self.clearJobs)
|
GUI.clearButton.clicked.connect(self.clearJobs)
|
||||||
GUI.fileButton.clicked.connect(self.selectFile)
|
GUI.fileButton.clicked.connect(self.selectFile)
|
||||||
GUI.directoryButton.clicked.connect(self.selectDir)
|
|
||||||
GUI.editorButton.clicked.connect(self.selectFileMetaEditor)
|
GUI.editorButton.clicked.connect(self.selectFileMetaEditor)
|
||||||
GUI.wikiButton.clicked.connect(self.openWiki)
|
GUI.wikiButton.clicked.connect(self.openWiki)
|
||||||
GUI.kofiButton.clicked.connect(self.openKofi)
|
|
||||||
GUI.convertButton.clicked.connect(self.convertStart)
|
GUI.convertButton.clicked.connect(self.convertStart)
|
||||||
GUI.gammaSlider.valueChanged.connect(self.changeGamma)
|
GUI.gammaSlider.valueChanged.connect(self.changeGamma)
|
||||||
GUI.gammaBox.stateChanged.connect(self.togglegammaBox)
|
GUI.gammaBox.stateChanged.connect(self.togglegammaBox)
|
||||||
GUI.croppingBox.stateChanged.connect(self.togglecroppingBox)
|
GUI.croppingBox.stateChanged.connect(self.togglecroppingBox)
|
||||||
GUI.croppingPowerSlider.valueChanged.connect(self.changeCroppingPower)
|
GUI.croppingPowerSlider.valueChanged.connect(self.changeCroppingPower)
|
||||||
GUI.jpegQualityBox.stateChanged.connect(self.togglejpegqualityBox)
|
|
||||||
GUI.webtoonBox.stateChanged.connect(self.togglewebtoonBox)
|
GUI.webtoonBox.stateChanged.connect(self.togglewebtoonBox)
|
||||||
GUI.qualityBox.stateChanged.connect(self.togglequalityBox)
|
GUI.qualityBox.stateChanged.connect(self.togglequalityBox)
|
||||||
GUI.mozJpegBox.stateChanged.connect(self.toggleImageFormatBox)
|
|
||||||
GUI.chunkSizeCheckBox.stateChanged.connect(self.togglechunkSizeCheckBox)
|
|
||||||
GUI.deviceBox.activated.connect(self.changeDevice)
|
GUI.deviceBox.activated.connect(self.changeDevice)
|
||||||
GUI.formatBox.activated.connect(self.changeFormat)
|
GUI.formatBox.activated.connect(self.changeFormat)
|
||||||
GUI.titleEdit.textChanged.connect(self.toggletitleEdit)
|
|
||||||
GUI.fileFusionBox.stateChanged.connect(self.togglefileFusionBox)
|
|
||||||
GUI.metadataTitleBox.stateChanged.connect(self.togglemetadataTitleBox)
|
|
||||||
GUI.jobList.itemDoubleClicked.connect(self.editSourceMetadata)
|
|
||||||
MW.progressBarTick.connect(self.updateProgressbar)
|
MW.progressBarTick.connect(self.updateProgressbar)
|
||||||
MW.modeConvert.connect(self.modeConvert)
|
MW.modeConvert.connect(self.modeConvert)
|
||||||
MW.addMessage.connect(self.addMessage)
|
MW.addMessage.connect(self.addMessage)
|
||||||
@@ -1445,11 +1146,6 @@ class KCCGUI(KCC_ui.Ui_mainWindow):
|
|||||||
if GUI.croppingPowerSlider.isEnabled():
|
if GUI.croppingPowerSlider.isEnabled():
|
||||||
GUI.croppingPowerSlider.setValue(int(self.options[option]))
|
GUI.croppingPowerSlider.setValue(int(self.options[option]))
|
||||||
self.changeCroppingPower(int(self.options[option]))
|
self.changeCroppingPower(int(self.options[option]))
|
||||||
GUI.preserveMarginBox.setValue(self.options.get('preserveMarginBox', 0))
|
|
||||||
elif str(option) == "jpegQuality":
|
|
||||||
GUI.jpegQualitySpinBox.setValue(int(self.options[option]))
|
|
||||||
elif str(option) == "chunkSizeBox":
|
|
||||||
GUI.chunkSizeBox.setValue(int(self.options[option]))
|
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
if getattr(GUI, option).isEnabled():
|
if getattr(GUI, option).isEnabled():
|
||||||
@@ -1485,17 +1181,15 @@ class KCCGUI_MetaEditor(KCC_ui_editor.Ui_editorDialog):
|
|||||||
self.editorWidget.setEnabled(True)
|
self.editorWidget.setEnabled(True)
|
||||||
self.okButton.setEnabled(True)
|
self.okButton.setEnabled(True)
|
||||||
self.statusLabel.setText('Separate authors with a comma.')
|
self.statusLabel.setText('Separate authors with a comma.')
|
||||||
for field in (self.seriesLine, self.volumeLine, self.numberLine, self.titleLine):
|
for field in (self.seriesLine, self.volumeLine, self.numberLine):
|
||||||
field.setText(self.parser.data[field.objectName().capitalize()[:-4]])
|
field.setText(self.parser.data[field.objectName().capitalize()[:-4]])
|
||||||
for field in (self.writerLine, self.pencillerLine, self.inkerLine, self.coloristLine):
|
for field in (self.writerLine, self.pencillerLine, self.inkerLine, self.coloristLine):
|
||||||
field.setText(', '.join(self.parser.data[field.objectName().capitalize()[:-4] + 's']))
|
field.setText(', '.join(self.parser.data[field.objectName().capitalize()[:-4] + 's']))
|
||||||
for field in (self.seriesLine, self.titleLine):
|
if self.seriesLine.text() == '':
|
||||||
if field.text() == '':
|
if file.endswith('.xml'):
|
||||||
path = Path(file)
|
self.seriesLine.setText(file.split('\\')[-2])
|
||||||
if file.endswith('.xml'):
|
else:
|
||||||
field.setText(path.parent.name)
|
self.seriesLine.setText(file.split('\\')[-1].split('/')[-1].split('.')[0])
|
||||||
else:
|
|
||||||
field.setText(path.stem)
|
|
||||||
|
|
||||||
def saveData(self):
|
def saveData(self):
|
||||||
for field in (self.volumeLine, self.numberLine):
|
for field in (self.volumeLine, self.numberLine):
|
||||||
@@ -1505,8 +1199,7 @@ class KCCGUI_MetaEditor(KCC_ui_editor.Ui_editorDialog):
|
|||||||
self.statusLabel.setText(field.objectName().capitalize()[:-4] + ' field must be a number.')
|
self.statusLabel.setText(field.objectName().capitalize()[:-4] + ' field must be a number.')
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
for field in (self.seriesLine, self.titleLine):
|
self.parser.data['Series'] = self.cleanData(self.seriesLine.text())
|
||||||
self.parser.data[field.objectName().capitalize()[:-4]] = self.cleanData(field.text())
|
|
||||||
for field in (self.writerLine, self.pencillerLine, self.inkerLine, self.coloristLine):
|
for field in (self.writerLine, self.pencillerLine, self.inkerLine, self.coloristLine):
|
||||||
values = self.cleanData(field.text()).split(',')
|
values = self.cleanData(field.text()).split(',')
|
||||||
tmpData = []
|
tmpData = []
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -3,7 +3,7 @@
|
|||||||
################################################################################
|
################################################################################
|
||||||
## Form generated from reading UI file 'MetaEditor.ui'
|
## Form generated from reading UI file 'MetaEditor.ui'
|
||||||
##
|
##
|
||||||
## Created by: Qt User Interface Compiler version 6.9.3
|
## Created by: Qt User Interface Compiler version 6.8.1
|
||||||
##
|
##
|
||||||
## WARNING! All changes made in this file will be lost when recompiling UI file!
|
## WARNING! All changes made in this file will be lost when recompiling UI file!
|
||||||
################################################################################
|
################################################################################
|
||||||
@@ -60,62 +60,52 @@ class Ui_editorDialog(object):
|
|||||||
self.label_3 = QLabel(self.editorWidget)
|
self.label_3 = QLabel(self.editorWidget)
|
||||||
self.label_3.setObjectName(u"label_3")
|
self.label_3.setObjectName(u"label_3")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.label_3, 3, 0, 1, 1)
|
self.gridLayout.addWidget(self.label_3, 2, 0, 1, 1)
|
||||||
|
|
||||||
self.numberLine = QLineEdit(self.editorWidget)
|
self.numberLine = QLineEdit(self.editorWidget)
|
||||||
self.numberLine.setObjectName(u"numberLine")
|
self.numberLine.setObjectName(u"numberLine")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.numberLine, 3, 1, 1, 1)
|
self.gridLayout.addWidget(self.numberLine, 2, 1, 1, 1)
|
||||||
|
|
||||||
self.label_4 = QLabel(self.editorWidget)
|
self.label_4 = QLabel(self.editorWidget)
|
||||||
self.label_4.setObjectName(u"label_4")
|
self.label_4.setObjectName(u"label_4")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.label_4, 4, 0, 1, 1)
|
self.gridLayout.addWidget(self.label_4, 3, 0, 1, 1)
|
||||||
|
|
||||||
self.writerLine = QLineEdit(self.editorWidget)
|
self.writerLine = QLineEdit(self.editorWidget)
|
||||||
self.writerLine.setObjectName(u"writerLine")
|
self.writerLine.setObjectName(u"writerLine")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.writerLine, 4, 1, 1, 1)
|
self.gridLayout.addWidget(self.writerLine, 3, 1, 1, 1)
|
||||||
|
|
||||||
self.label_5 = QLabel(self.editorWidget)
|
self.label_5 = QLabel(self.editorWidget)
|
||||||
self.label_5.setObjectName(u"label_5")
|
self.label_5.setObjectName(u"label_5")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.label_5, 5, 0, 1, 1)
|
self.gridLayout.addWidget(self.label_5, 4, 0, 1, 1)
|
||||||
|
|
||||||
self.pencillerLine = QLineEdit(self.editorWidget)
|
self.pencillerLine = QLineEdit(self.editorWidget)
|
||||||
self.pencillerLine.setObjectName(u"pencillerLine")
|
self.pencillerLine.setObjectName(u"pencillerLine")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.pencillerLine, 5, 1, 1, 1)
|
self.gridLayout.addWidget(self.pencillerLine, 4, 1, 1, 1)
|
||||||
|
|
||||||
self.label_6 = QLabel(self.editorWidget)
|
self.label_6 = QLabel(self.editorWidget)
|
||||||
self.label_6.setObjectName(u"label_6")
|
self.label_6.setObjectName(u"label_6")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.label_6, 6, 0, 1, 1)
|
self.gridLayout.addWidget(self.label_6, 5, 0, 1, 1)
|
||||||
|
|
||||||
self.inkerLine = QLineEdit(self.editorWidget)
|
self.inkerLine = QLineEdit(self.editorWidget)
|
||||||
self.inkerLine.setObjectName(u"inkerLine")
|
self.inkerLine.setObjectName(u"inkerLine")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.inkerLine, 6, 1, 1, 1)
|
self.gridLayout.addWidget(self.inkerLine, 5, 1, 1, 1)
|
||||||
|
|
||||||
self.label_7 = QLabel(self.editorWidget)
|
self.label_7 = QLabel(self.editorWidget)
|
||||||
self.label_7.setObjectName(u"label_7")
|
self.label_7.setObjectName(u"label_7")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.label_7, 7, 0, 1, 1)
|
self.gridLayout.addWidget(self.label_7, 6, 0, 1, 1)
|
||||||
|
|
||||||
self.coloristLine = QLineEdit(self.editorWidget)
|
self.coloristLine = QLineEdit(self.editorWidget)
|
||||||
self.coloristLine.setObjectName(u"coloristLine")
|
self.coloristLine.setObjectName(u"coloristLine")
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.coloristLine, 7, 1, 1, 1)
|
self.gridLayout.addWidget(self.coloristLine, 6, 1, 1, 1)
|
||||||
|
|
||||||
self.label_8 = QLabel(self.editorWidget)
|
|
||||||
self.label_8.setObjectName(u"label_8")
|
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.label_8, 2, 0, 1, 1)
|
|
||||||
|
|
||||||
self.titleLine = QLineEdit(self.editorWidget)
|
|
||||||
self.titleLine.setObjectName(u"titleLine")
|
|
||||||
|
|
||||||
self.gridLayout.addWidget(self.titleLine, 2, 1, 1, 1)
|
|
||||||
|
|
||||||
|
|
||||||
self.verticalLayout.addWidget(self.editorWidget)
|
self.verticalLayout.addWidget(self.editorWidget)
|
||||||
@@ -156,15 +146,6 @@ class Ui_editorDialog(object):
|
|||||||
|
|
||||||
self.verticalLayout.addWidget(self.optionWidget)
|
self.verticalLayout.addWidget(self.optionWidget)
|
||||||
|
|
||||||
QWidget.setTabOrder(self.seriesLine, self.volumeLine)
|
|
||||||
QWidget.setTabOrder(self.volumeLine, self.titleLine)
|
|
||||||
QWidget.setTabOrder(self.titleLine, self.numberLine)
|
|
||||||
QWidget.setTabOrder(self.numberLine, self.writerLine)
|
|
||||||
QWidget.setTabOrder(self.writerLine, self.pencillerLine)
|
|
||||||
QWidget.setTabOrder(self.pencillerLine, self.inkerLine)
|
|
||||||
QWidget.setTabOrder(self.inkerLine, self.coloristLine)
|
|
||||||
QWidget.setTabOrder(self.coloristLine, self.okButton)
|
|
||||||
QWidget.setTabOrder(self.okButton, self.cancelButton)
|
|
||||||
|
|
||||||
self.retranslateUi(editorDialog)
|
self.retranslateUi(editorDialog)
|
||||||
|
|
||||||
@@ -180,7 +161,6 @@ class Ui_editorDialog(object):
|
|||||||
self.label_5.setText(QCoreApplication.translate("editorDialog", u"Penciller:", None))
|
self.label_5.setText(QCoreApplication.translate("editorDialog", u"Penciller:", None))
|
||||||
self.label_6.setText(QCoreApplication.translate("editorDialog", u"Inker:", None))
|
self.label_6.setText(QCoreApplication.translate("editorDialog", u"Inker:", None))
|
||||||
self.label_7.setText(QCoreApplication.translate("editorDialog", u"Colorist:", None))
|
self.label_7.setText(QCoreApplication.translate("editorDialog", u"Colorist:", None))
|
||||||
self.label_8.setText(QCoreApplication.translate("editorDialog", u"Title:", None))
|
|
||||||
self.statusLabel.setText("")
|
self.statusLabel.setText("")
|
||||||
self.okButton.setText(QCoreApplication.translate("editorDialog", u"Save", None))
|
self.okButton.setText(QCoreApplication.translate("editorDialog", u"Save", None))
|
||||||
self.cancelButton.setText(QCoreApplication.translate("editorDialog", u"Cancel", None))
|
self.cancelButton.setText(QCoreApplication.translate("editorDialog", u"Cancel", None))
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
__version__ = '9.5.0'
|
__version__ = '7.1.2'
|
||||||
__license__ = 'ISC'
|
__license__ = 'ISC'
|
||||||
__copyright__ = '2012-2022, Ciro Mattia Gonano <ciromattia@gmail.com>, Pawel Jastrzebski <pawelj@iosphe.re>, darodi'
|
__copyright__ = '2012-2022, Ciro Mattia Gonano <ciromattia@gmail.com>, Pawel Jastrzebski <pawelj@iosphe.re>, darodi'
|
||||||
__docformat__ = 'restructuredtext en'
|
__docformat__ = 'restructuredtext en'
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -18,17 +18,13 @@
|
|||||||
# PERFORMANCE OF THIS SOFTWARE.
|
# PERFORMANCE OF THIS SOFTWARE.
|
||||||
#
|
#
|
||||||
|
|
||||||
import math
|
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
from argparse import ArgumentParser
|
from argparse import ArgumentParser
|
||||||
from shutil import rmtree
|
from shutil import rmtree, copytree, move
|
||||||
from multiprocessing import Pool
|
from multiprocessing import Pool
|
||||||
from PIL import Image, ImageChops, ImageOps, ImageDraw, ImageFilter, ImageFile
|
from PIL import Image, ImageChops, ImageOps, ImageDraw
|
||||||
from PIL.Image import Dither
|
from .shared import getImageFileName, walkLevel, walkSort, sanitizeTrace
|
||||||
from .shared import dot_clean, getImageFileName, walkLevel, walkSort, sanitizeTrace
|
|
||||||
|
|
||||||
ImageFile.LOAD_TRUNCATED_IMAGES = True
|
|
||||||
|
|
||||||
|
|
||||||
def mergeDirectoryTick(output):
|
def mergeDirectoryTick(output):
|
||||||
@@ -48,7 +44,6 @@ def mergeDirectory(work):
|
|||||||
imagesValid = []
|
imagesValid = []
|
||||||
sizes = []
|
sizes = []
|
||||||
targetHeight = 0
|
targetHeight = 0
|
||||||
dot_clean(directory)
|
|
||||||
for root, _, files in walkLevel(directory, 0):
|
for root, _, files in walkLevel(directory, 0):
|
||||||
for name in files:
|
for name in files:
|
||||||
if getImageFileName(name) is not None:
|
if getImageFileName(name) is not None:
|
||||||
@@ -62,19 +57,18 @@ def mergeDirectory(work):
|
|||||||
imagesValid.append(i[0])
|
imagesValid.append(i[0])
|
||||||
# Silently drop directories that contain too many images
|
# Silently drop directories that contain too many images
|
||||||
# 131072 = GIMP_MAX_IMAGE_SIZE / 4
|
# 131072 = GIMP_MAX_IMAGE_SIZE / 4
|
||||||
if targetHeight > 131072 * 4:
|
if targetHeight > 131072:
|
||||||
raise RuntimeError(f'Image too tall at {targetHeight} pixels. {targetWidth} pixels wide. Try using separate chapter folders or file fusion.')
|
return None
|
||||||
result = Image.new('RGB', (targetWidth, targetHeight))
|
result = Image.new('RGB', (targetWidth, targetHeight))
|
||||||
y = 0
|
y = 0
|
||||||
for i in imagesValid:
|
for i in imagesValid:
|
||||||
with Image.open(i) as img:
|
img = Image.open(i).convert('RGB')
|
||||||
img = img.convert('RGB')
|
if img.size[0] < targetWidth or img.size[0] > targetWidth:
|
||||||
if img.size[0] < targetWidth or img.size[0] > targetWidth:
|
widthPercent = (targetWidth / float(img.size[0]))
|
||||||
widthPercent = (targetWidth / float(img.size[0]))
|
heightSize = int((float(img.size[1]) * float(widthPercent)))
|
||||||
heightSize = int((float(img.size[1]) * float(widthPercent)))
|
img = ImageOps.fit(img, (targetWidth, heightSize), method=Image.BICUBIC, centering=(0.5, 0.5))
|
||||||
img = ImageOps.fit(img, (targetWidth, heightSize), method=Image.BICUBIC, centering=(0.5, 0.5))
|
result.paste(img, (0, y))
|
||||||
result.paste(img, (0, y))
|
y += img.size[1]
|
||||||
y += img.size[1]
|
|
||||||
os.remove(i)
|
os.remove(i)
|
||||||
savePath = os.path.split(imagesValid[0])
|
savePath = os.path.split(imagesValid[0])
|
||||||
result.save(os.path.join(savePath[0], os.path.splitext(savePath[1])[0] + '.png'), 'PNG')
|
result.save(os.path.join(savePath[0], os.path.splitext(savePath[1])[0] + '.png'), 'PNG')
|
||||||
@@ -106,11 +100,7 @@ def splitImage(work):
|
|||||||
Image.warnings.simplefilter('error', Image.DecompressionBombWarning)
|
Image.warnings.simplefilter('error', Image.DecompressionBombWarning)
|
||||||
Image.MAX_IMAGE_PIXELS = 1000000000
|
Image.MAX_IMAGE_PIXELS = 1000000000
|
||||||
imgOrg = Image.open(filePath).convert('RGB')
|
imgOrg = Image.open(filePath).convert('RGB')
|
||||||
# I experimented with custom vertical edge kernel [-1, 2, -1] but got poor results
|
imgProcess = Image.open(filePath).convert('1')
|
||||||
imgEdges = Image.open(filePath).convert('L').filter(ImageFilter.FIND_EDGES)
|
|
||||||
# threshold of 8 is too high. 5 is too low.
|
|
||||||
imgProcess = imgEdges.point(lambda p: 255 if p > 6 else 0).convert('1', dither=Dither.NONE)
|
|
||||||
|
|
||||||
widthImg, heightImg = imgOrg.size
|
widthImg, heightImg = imgOrg.size
|
||||||
if heightImg > opt.height:
|
if heightImg > opt.height:
|
||||||
if opt.debug:
|
if opt.debug:
|
||||||
@@ -121,71 +111,47 @@ def splitImage(work):
|
|||||||
yWork = 0
|
yWork = 0
|
||||||
panelDetected = False
|
panelDetected = False
|
||||||
panels = []
|
panels = []
|
||||||
# check git history for how these constant values changed
|
|
||||||
h_pad = int(widthImg / 20)
|
|
||||||
v_pad = int(widthImg / 80)
|
|
||||||
if v_pad % 2:
|
|
||||||
v_pad += 1
|
|
||||||
while yWork < heightImg:
|
while yWork < heightImg:
|
||||||
tmpImg = imgProcess.crop((h_pad, yWork, widthImg - h_pad, yWork + v_pad))
|
tmpImg = imgProcess.crop((4, yWork, widthImg-4, yWork + 4))
|
||||||
solid = detectSolid(tmpImg)
|
solid = detectSolid(tmpImg)
|
||||||
if not solid and not panelDetected:
|
if not solid and not panelDetected:
|
||||||
panelDetected = True
|
panelDetected = True
|
||||||
panelY1 = yWork
|
panelY1 = yWork - 2
|
||||||
if heightImg - yWork <= (v_pad // 2):
|
if heightImg - yWork <= 5:
|
||||||
if not solid and panelDetected:
|
if not solid and panelDetected:
|
||||||
panelY2 = heightImg
|
panelY2 = heightImg
|
||||||
panelDetected = False
|
panelDetected = False
|
||||||
panels.append((panelY1, panelY2, panelY2 - panelY1))
|
panels.append((panelY1, panelY2, panelY2 - panelY1))
|
||||||
if solid and panelDetected:
|
if solid and panelDetected:
|
||||||
panelDetected = False
|
panelDetected = False
|
||||||
panelY2 = yWork
|
panelY2 = yWork + 6
|
||||||
# skip short panel at start
|
|
||||||
if panelY1 < v_pad * 2 and panelY2 - panelY1 < v_pad * 2:
|
|
||||||
continue
|
|
||||||
panels.append((panelY1, panelY2, panelY2 - panelY1))
|
panels.append((panelY1, panelY2, panelY2 - panelY1))
|
||||||
yWork += v_pad // 2
|
yWork += 5
|
||||||
|
|
||||||
max_width = 1072
|
|
||||||
virtual_width = min((max_width, opt.width, widthImg))
|
|
||||||
if opt.width > max_width:
|
|
||||||
virtual_height = int(opt.height/max_width*virtual_width)
|
|
||||||
else:
|
|
||||||
virtual_height = int(opt.height/opt.width*virtual_width)
|
|
||||||
opt.height = virtual_height
|
|
||||||
|
|
||||||
# Split too big panels
|
# Split too big panels
|
||||||
panelsProcessed = []
|
panelsProcessed = []
|
||||||
for panel in panels:
|
for panel in panels:
|
||||||
# 1.52 too high
|
|
||||||
if panel[2] <= opt.height * 1.5:
|
if panel[2] <= opt.height * 1.5:
|
||||||
panelsProcessed.append(panel)
|
panelsProcessed.append(panel)
|
||||||
elif panel[2] <= opt.height * 2:
|
elif panel[2] < opt.height * 2:
|
||||||
diff = panel[2] - opt.height
|
diff = panel[2] - opt.height
|
||||||
panelsProcessed.append((panel[0], panel[1] - diff, opt.height))
|
panelsProcessed.append((panel[0], panel[1] - diff, opt.height))
|
||||||
panelsProcessed.append((panel[1] - opt.height, panel[1], opt.height))
|
panelsProcessed.append((panel[1] - opt.height, panel[1], opt.height))
|
||||||
else:
|
else:
|
||||||
# split super long panels with overlap
|
parts = round(panel[2] / opt.height)
|
||||||
parts = math.ceil(panel[2] / opt.height)
|
|
||||||
diff = panel[2] // parts
|
diff = panel[2] // parts
|
||||||
panelsProcessed.append((panel[0], panel[0] + opt.height, opt.height))
|
for x in range(0, parts):
|
||||||
for x in range(1, parts - 1):
|
panelsProcessed.append((panel[0] + (x * diff), panel[1] - ((parts - x - 1) * diff), diff))
|
||||||
start = panel[0] + (x * diff)
|
|
||||||
panelsProcessed.append((start, start + opt.height, opt.height))
|
|
||||||
panelsProcessed.append((panel[1] - opt.height, panel[1], opt.height))
|
|
||||||
|
|
||||||
if opt.debug:
|
if opt.debug:
|
||||||
for panel in panelsProcessed:
|
for panel in panelsProcessed:
|
||||||
draw.rectangle(((0, panel[0]), (widthImg, panel[1])), (0, 255, 0, 128), (0, 0, 255, 255))
|
draw.rectangle(((0, panel[0]), (widthImg, panel[1])), (0, 255, 0, 128), (0, 0, 255, 255))
|
||||||
debugImage = Image.alpha_composite(imgOrg.convert(mode='RGBA'), drawImg)
|
debugImage = Image.alpha_composite(imgOrg.convert(mode='RGBA'), drawImg)
|
||||||
# debugImage.show()
|
|
||||||
debugImage.save(os.path.join(path, os.path.splitext(name)[0] + '-debug.png'), 'PNG')
|
debugImage.save(os.path.join(path, os.path.splitext(name)[0] + '-debug.png'), 'PNG')
|
||||||
|
|
||||||
# Create virtual pages
|
# Create virtual pages
|
||||||
pages = []
|
pages = []
|
||||||
currentPage = []
|
currentPage = []
|
||||||
# TODO: 1.25 way too high, 1.1 too high, 1.05 slightly too high(?), optimized for 2 page landscape reading
|
|
||||||
# opt.height = max_height = virtual_height * 1.00
|
|
||||||
pageLeft = opt.height
|
pageLeft = opt.height
|
||||||
panelNumber = 0
|
panelNumber = 0
|
||||||
for panel in panelsProcessed:
|
for panel in panelsProcessed:
|
||||||
@@ -215,14 +181,14 @@ def splitImage(work):
|
|||||||
panelImg = imgOrg.crop((0, panelsProcessed[panel][0], widthImg, panelsProcessed[panel][1]))
|
panelImg = imgOrg.crop((0, panelsProcessed[panel][0], widthImg, panelsProcessed[panel][1]))
|
||||||
newPage.paste(panelImg, (0, targetHeight))
|
newPage.paste(panelImg, (0, targetHeight))
|
||||||
targetHeight += panelsProcessed[panel][2]
|
targetHeight += panelsProcessed[panel][2]
|
||||||
newPage.save(os.path.join(path, os.path.splitext(name)[0] + '-' + str(pageNumber).zfill(4) + '.png'), 'PNG')
|
newPage.save(os.path.join(path, os.path.splitext(name)[0] + '-' + str(pageNumber) + '.png'), 'PNG')
|
||||||
pageNumber += 1
|
pageNumber += 1
|
||||||
os.remove(filePath)
|
os.remove(filePath)
|
||||||
except Exception:
|
except Exception:
|
||||||
return str(sys.exc_info()[1]), sanitizeTrace(sys.exc_info()[2])
|
return str(sys.exc_info()[1]), sanitizeTrace(sys.exc_info()[2])
|
||||||
|
|
||||||
|
|
||||||
def main(argv=None, job_progress='', qtgui=None):
|
def main(argv=None, qtgui=None):
|
||||||
global args, GUI, splitWorkerPool, splitWorkerOutput, mergeWorkerPool, mergeWorkerOutput
|
global args, GUI, splitWorkerPool, splitWorkerOutput, mergeWorkerPool, mergeWorkerOutput
|
||||||
parser = ArgumentParser(prog="kcc-c2p", usage="kcc-c2p [options] [input]", add_help=False)
|
parser = ArgumentParser(prog="kcc-c2p", usage="kcc-c2p [options] [input]", add_help=False)
|
||||||
|
|
||||||
@@ -234,8 +200,6 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
" with spaces.")
|
" with spaces.")
|
||||||
main_options.add_argument("-y", "--height", type=int, dest="height", default=0,
|
main_options.add_argument("-y", "--height", type=int, dest="height", default=0,
|
||||||
help="Height of the target device screen")
|
help="Height of the target device screen")
|
||||||
main_options.add_argument("-x", "--width", type=int, dest="width", default=0,
|
|
||||||
help="Width of the target device screen")
|
|
||||||
main_options.add_argument("-i", "--in-place", action="store_true", dest="inPlace", default=False,
|
main_options.add_argument("-i", "--in-place", action="store_true", dest="inPlace", default=False,
|
||||||
help="Overwrite source directory")
|
help="Overwrite source directory")
|
||||||
main_options.add_argument("-m", "--merge", action="store_true", dest="merge", default=False,
|
main_options.add_argument("-m", "--merge", action="store_true", dest="merge", default=False,
|
||||||
@@ -254,14 +218,16 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
return 1
|
return 1
|
||||||
if args.height > 0:
|
if args.height > 0:
|
||||||
for sourceDir in args.input:
|
for sourceDir in args.input:
|
||||||
targetDir = sourceDir
|
targetDir = sourceDir + "-Splitted"
|
||||||
if os.path.isdir(sourceDir):
|
if os.path.isdir(sourceDir):
|
||||||
|
rmtree(targetDir, True)
|
||||||
|
copytree(sourceDir, targetDir)
|
||||||
work = []
|
work = []
|
||||||
pagenumber = 1
|
pagenumber = 1
|
||||||
splitWorkerOutput = []
|
splitWorkerOutput = []
|
||||||
splitWorkerPool = Pool(maxtasksperchild=10)
|
splitWorkerPool = Pool(maxtasksperchild=10)
|
||||||
if args.merge:
|
if args.merge:
|
||||||
print(f"{job_progress}Merging images...")
|
print("Merging images...")
|
||||||
directoryNumer = 1
|
directoryNumer = 1
|
||||||
mergeWork = []
|
mergeWork = []
|
||||||
mergeWorkerOutput = []
|
mergeWorkerOutput = []
|
||||||
@@ -273,7 +239,7 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
directoryNumer += 1
|
directoryNumer += 1
|
||||||
mergeWork.append([os.path.join(root, directory)])
|
mergeWork.append([os.path.join(root, directory)])
|
||||||
if GUI:
|
if GUI:
|
||||||
GUI.progressBarTick.emit(f'{job_progress}Combining images')
|
GUI.progressBarTick.emit('Combining images')
|
||||||
GUI.progressBarTick.emit(str(directoryNumer))
|
GUI.progressBarTick.emit(str(directoryNumer))
|
||||||
for i in mergeWork:
|
for i in mergeWork:
|
||||||
mergeWorkerPool.apply_async(func=mergeDirectory, args=(i, ), callback=mergeDirectoryTick)
|
mergeWorkerPool.apply_async(func=mergeDirectory, args=(i, ), callback=mergeDirectoryTick)
|
||||||
@@ -286,8 +252,7 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
rmtree(targetDir, True)
|
rmtree(targetDir, True)
|
||||||
raise RuntimeError("One of workers crashed. Cause: " + mergeWorkerOutput[0][0],
|
raise RuntimeError("One of workers crashed. Cause: " + mergeWorkerOutput[0][0],
|
||||||
mergeWorkerOutput[0][1])
|
mergeWorkerOutput[0][1])
|
||||||
print(f"{job_progress}Splitting images...")
|
print("Splitting images...")
|
||||||
dot_clean(targetDir)
|
|
||||||
for root, _, files in os.walk(targetDir, False):
|
for root, _, files in os.walk(targetDir, False):
|
||||||
for name in files:
|
for name in files:
|
||||||
if getImageFileName(name) is not None:
|
if getImageFileName(name) is not None:
|
||||||
@@ -296,7 +261,7 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
else:
|
else:
|
||||||
os.remove(os.path.join(root, name))
|
os.remove(os.path.join(root, name))
|
||||||
if GUI:
|
if GUI:
|
||||||
GUI.progressBarTick.emit(f'{job_progress}Splitting images')
|
GUI.progressBarTick.emit('Splitting images')
|
||||||
GUI.progressBarTick.emit(str(pagenumber))
|
GUI.progressBarTick.emit(str(pagenumber))
|
||||||
GUI.progressBarTick.emit('tick')
|
GUI.progressBarTick.emit('tick')
|
||||||
if len(work) > 0:
|
if len(work) > 0:
|
||||||
@@ -304,7 +269,6 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
splitWorkerPool.apply_async(func=splitImage, args=(i, ), callback=splitImageTick)
|
splitWorkerPool.apply_async(func=splitImage, args=(i, ), callback=splitImageTick)
|
||||||
splitWorkerPool.close()
|
splitWorkerPool.close()
|
||||||
splitWorkerPool.join()
|
splitWorkerPool.join()
|
||||||
dot_clean(targetDir)
|
|
||||||
if GUI and not GUI.conversionAlive:
|
if GUI and not GUI.conversionAlive:
|
||||||
rmtree(targetDir, True)
|
rmtree(targetDir, True)
|
||||||
raise UserWarning("Conversion interrupted.")
|
raise UserWarning("Conversion interrupted.")
|
||||||
@@ -312,9 +276,12 @@ def main(argv=None, job_progress='', qtgui=None):
|
|||||||
rmtree(targetDir, True)
|
rmtree(targetDir, True)
|
||||||
raise RuntimeError("One of workers crashed. Cause: " + splitWorkerOutput[0][0],
|
raise RuntimeError("One of workers crashed. Cause: " + splitWorkerOutput[0][0],
|
||||||
splitWorkerOutput[0][1])
|
splitWorkerOutput[0][1])
|
||||||
|
if args.inPlace:
|
||||||
|
rmtree(sourceDir)
|
||||||
|
move(targetDir, sourceDir)
|
||||||
else:
|
else:
|
||||||
rmtree(targetDir, True)
|
rmtree(targetDir, True)
|
||||||
raise UserWarning("C2P: Source directory is empty.")
|
raise UserWarning("Source directory is empty.")
|
||||||
else:
|
else:
|
||||||
raise UserWarning("Provided input is not a directory.")
|
raise UserWarning("Provided input is not a directory.")
|
||||||
else:
|
else:
|
||||||
|
|||||||
@@ -18,19 +18,16 @@
|
|||||||
# PERFORMANCE OF THIS SOFTWARE.
|
# PERFORMANCE OF THIS SOFTWARE.
|
||||||
#
|
#
|
||||||
|
|
||||||
from functools import cached_property, lru_cache
|
from functools import cached_property
|
||||||
import os
|
import os
|
||||||
from pathlib import Path
|
|
||||||
import platform
|
import platform
|
||||||
import distro
|
import distro
|
||||||
from subprocess import STDOUT, PIPE, CalledProcessError
|
from subprocess import STDOUT, PIPE, CalledProcessError
|
||||||
from xml.dom.minidom import parseString
|
from xml.dom.minidom import parseString
|
||||||
from xml.parsers.expat import ExpatError
|
from xml.parsers.expat import ExpatError
|
||||||
from .shared import IMAGE_TYPES, subprocess_run
|
from .shared import subprocess_run
|
||||||
|
|
||||||
EXTRACTION_ERROR = 'Failed to extract archive. Try extracting file outside of KCC.'
|
EXTRACTION_ERROR = 'Failed to extract archive. Try extracting file outside of KCC.'
|
||||||
SEVENZIP = '7zz' if platform.system() == 'Darwin' else '7z'
|
|
||||||
TAR = 'bsdtar' if platform.system() == 'Linux' else 'tar'
|
|
||||||
|
|
||||||
|
|
||||||
class ComicArchive:
|
class ComicArchive:
|
||||||
@@ -38,22 +35,21 @@ class ComicArchive:
|
|||||||
self.filepath = filepath
|
self.filepath = filepath
|
||||||
if not os.path.isfile(self.filepath):
|
if not os.path.isfile(self.filepath):
|
||||||
raise OSError('File not found.')
|
raise OSError('File not found.')
|
||||||
self.dirname, self.basename = os.path.split(filepath)
|
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def type(self):
|
def type(self):
|
||||||
extraction_commands = [
|
extraction_commands = [
|
||||||
[SEVENZIP, 'l', '-y', '-p1', self.basename],
|
['7z', 'l', '-y', '-p1', self.filepath],
|
||||||
]
|
]
|
||||||
|
|
||||||
if distro.id() == 'fedora' or distro.like() == 'fedora':
|
if distro.id() == 'fedora' or distro.like() == 'fedora':
|
||||||
extraction_commands.append(
|
extraction_commands.append(
|
||||||
['unrar', 'l', '-y', '-p1', self.basename],
|
['unrar', 'l', '-y', '-p1', self.filepath],
|
||||||
)
|
)
|
||||||
|
|
||||||
for cmd in extraction_commands:
|
for cmd in extraction_commands:
|
||||||
try:
|
try:
|
||||||
process = subprocess_run(cmd, capture_output=True, check=True, cwd=self.dirname)
|
process = subprocess_run(cmd, capture_output=True, check=True)
|
||||||
for line in process.stdout.splitlines():
|
for line in process.stdout.splitlines():
|
||||||
if b'Type =' in line:
|
if b'Type =' in line:
|
||||||
return line.rstrip().decode().split(' = ')[1].upper()
|
return line.rstrip().decode().split(' = ')[1].upper()
|
||||||
@@ -67,33 +63,28 @@ class ComicArchive:
|
|||||||
def extract(self, targetdir):
|
def extract(self, targetdir):
|
||||||
if not os.path.isdir(targetdir):
|
if not os.path.isdir(targetdir):
|
||||||
raise OSError('Target directory doesn\'t exist.')
|
raise OSError('Target directory doesn\'t exist.')
|
||||||
|
|
||||||
if Path(self.basename).suffix.lower() in IMAGE_TYPES:
|
|
||||||
raise UserWarning('Put images into folder and drag and drop folder into KCC window.')
|
|
||||||
|
|
||||||
missing = []
|
missing = []
|
||||||
|
|
||||||
extraction_commands = [
|
extraction_commands = [
|
||||||
[TAR, '--exclude', '__MACOSX', '--exclude', '.DS_Store', '--exclude', 'thumbs.db', '--exclude', 'Thumbs.db', '-xf', self.basename, '-C', targetdir],
|
['tar', '--exclude', '__MACOSX', '--exclude', '.DS_Store', '--exclude', 'thumbs.db', '--exclude', 'Thumbs.db', '-xf', self.filepath, '-C', targetdir],
|
||||||
[SEVENZIP, 'x', '-y', '-xr!__MACOSX', '-xr!.DS_Store', '-xr!thumbs.db', '-xr!Thumbs.db', '-o' + targetdir, self.basename],
|
['7z', 'x', '-y', '-xr!__MACOSX', '-xr!.DS_Store', '-xr!thumbs.db', '-xr!Thumbs.db', '-o' + targetdir, self.filepath],
|
||||||
]
|
]
|
||||||
|
|
||||||
if platform.system() == 'Darwin':
|
if platform.system() == 'Darwin':
|
||||||
extraction_commands.append(
|
extraction_commands.append(
|
||||||
['unar', self.basename, '-D', '-f', '-o', targetdir]
|
['unar', self.filepath, '-f', '-o', targetdir]
|
||||||
)
|
)
|
||||||
|
|
||||||
extraction_commands.reverse()
|
|
||||||
|
|
||||||
if distro.id() == 'fedora' or distro.like() == 'fedora':
|
if distro.id() == 'fedora' or distro.like() == 'fedora':
|
||||||
extraction_commands.append(
|
extraction_commands.append(
|
||||||
['unrar', 'x', '-y', '-x__MACOSX', '-x.DS_Store', '-xthumbs.db', '-xThumbs.db', self.basename, targetdir]
|
['unrar', 'x', '-y', '-x__MACOSX', '-x.DS_Store', '-xthumbs.db', '-xThumbs.db', self.filepath, targetdir]
|
||||||
)
|
)
|
||||||
|
|
||||||
for cmd in extraction_commands:
|
for cmd in extraction_commands:
|
||||||
try:
|
try:
|
||||||
subprocess_run(cmd, capture_output=True, check=True, cwd=self.dirname)
|
subprocess_run(cmd, capture_output=True, check=True)
|
||||||
return targetdir
|
return targetdir
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
missing.append(cmd[0])
|
missing.append(cmd[0])
|
||||||
except CalledProcessError:
|
except CalledProcessError:
|
||||||
@@ -107,30 +98,17 @@ class ComicArchive:
|
|||||||
def addFile(self, sourcefile):
|
def addFile(self, sourcefile):
|
||||||
if self.type in ['RAR', 'RAR5']:
|
if self.type in ['RAR', 'RAR5']:
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
process = subprocess_run([SEVENZIP, 'a', '-y', self.basename, sourcefile],
|
process = subprocess_run(['7z', 'a', '-y', self.filepath, sourcefile],
|
||||||
stdout=PIPE, stderr=STDOUT, cwd=self.dirname)
|
stdout=PIPE, stderr=STDOUT)
|
||||||
if process.returncode != 0:
|
if process.returncode != 0:
|
||||||
raise OSError('Failed to add the file.')
|
raise OSError('Failed to add the file.')
|
||||||
|
|
||||||
def extractMetadata(self):
|
def extractMetadata(self):
|
||||||
process = subprocess_run([SEVENZIP, 'x', '-y', '-so', self.basename, 'ComicInfo.xml'],
|
process = subprocess_run(['7z', 'x', '-y', '-so', self.filepath, 'ComicInfo.xml'],
|
||||||
stdout=PIPE, stderr=STDOUT, cwd=self.dirname)
|
stdout=PIPE, stderr=STDOUT)
|
||||||
if process.returncode != 0:
|
if process.returncode != 0:
|
||||||
raise OSError(EXTRACTION_ERROR)
|
raise OSError(EXTRACTION_ERROR)
|
||||||
try:
|
try:
|
||||||
return parseString(process.stdout)
|
return parseString(process.stdout)
|
||||||
except ExpatError:
|
except ExpatError:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@lru_cache
|
|
||||||
def available_archive_tools():
|
|
||||||
available = []
|
|
||||||
|
|
||||||
for tool in [TAR, SEVENZIP, 'unar', 'unrar']:
|
|
||||||
try:
|
|
||||||
subprocess_run([tool], stdout=PIPE, stderr=STDOUT)
|
|
||||||
available.append(tool)
|
|
||||||
except (FileNotFoundError, CalledProcessError):
|
|
||||||
pass
|
|
||||||
|
|
||||||
return available
|
|
||||||
|
|||||||
@@ -1,28 +0,0 @@
|
|||||||
def threshold_from_power(power):
|
|
||||||
return 240-(power*64)
|
|
||||||
|
|
||||||
|
|
||||||
'''
|
|
||||||
Groups close values together
|
|
||||||
'''
|
|
||||||
def group_close_values(vals, max_dist_tolerated):
|
|
||||||
groups = []
|
|
||||||
|
|
||||||
group_start = -1
|
|
||||||
group_end = 0
|
|
||||||
for i in range(len(vals)):
|
|
||||||
dist = vals[i] - group_end
|
|
||||||
if group_start == -1:
|
|
||||||
group_start = vals[i]
|
|
||||||
group_end = vals[i]
|
|
||||||
elif dist <= max_dist_tolerated:
|
|
||||||
group_end = vals[i]
|
|
||||||
else:
|
|
||||||
groups.append((group_start, group_end))
|
|
||||||
group_start = -1
|
|
||||||
group_end = -1
|
|
||||||
|
|
||||||
if group_start != -1:
|
|
||||||
groups.append((group_start, group_end))
|
|
||||||
|
|
||||||
return groups
|
|
||||||
@@ -20,18 +20,12 @@
|
|||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
import io
|
import io
|
||||||
import os
|
import os
|
||||||
import numpy as np
|
|
||||||
from pathlib import Path
|
|
||||||
from functools import cached_property
|
|
||||||
import mozjpeg_lossless_optimization
|
import mozjpeg_lossless_optimization
|
||||||
from PIL import Image, ImageOps, ImageFile, ImageChops, ImageDraw
|
from PIL import Image, ImageOps, ImageStat, ImageChops, ImageFilter
|
||||||
|
from .shared import md5Checksum
|
||||||
from .rainbow_artifacts_eraser import erase_rainbow_artifacts
|
|
||||||
from .page_number_crop_alg import get_bbox_crop_margin_page_number, get_bbox_crop_margin
|
from .page_number_crop_alg import get_bbox_crop_margin_page_number, get_bbox_crop_margin
|
||||||
from .inter_panel_crop_alg import crop_empty_inter_panel
|
|
||||||
|
|
||||||
AUTO_CROP_THRESHOLD = 0.015
|
AUTO_CROP_THRESHOLD = 0.015
|
||||||
ImageFile.LOAD_TRUNCATED_IMAGES = True
|
|
||||||
|
|
||||||
|
|
||||||
class ProfileData:
|
class ProfileData:
|
||||||
@@ -86,28 +80,20 @@ class ProfileData:
|
|||||||
]
|
]
|
||||||
|
|
||||||
ProfilesKindleEBOK = {
|
ProfilesKindleEBOK = {
|
||||||
|
'K1': ("Kindle 1", (600, 670), Palette4, 1.8),
|
||||||
|
'K2': ("Kindle 2", (600, 670), Palette15, 1.8),
|
||||||
|
'KDX': ("Kindle DX/DXG", (824, 1000), Palette16, 1.8),
|
||||||
|
'K34': ("Kindle Keyboard/Touch", (600, 800), Palette16, 1.8),
|
||||||
|
'K578': ("Kindle", (600, 800), Palette16, 1.8),
|
||||||
|
'KPW': ("Kindle Paperwhite 1/2", (758, 1024), Palette16, 1.8),
|
||||||
|
'KV': ("Kindle Paperwhite 3/4/Voyage/Oasis", (1072, 1448), Palette16, 1.8),
|
||||||
}
|
}
|
||||||
|
|
||||||
ProfilesKindlePDOC = {
|
ProfilesKindlePDOC = {
|
||||||
'K1': ("Kindle 1", (600, 670), Palette4, 1.0),
|
'KO': ("Kindle Oasis 2/3/Paperwhite 12/Colorsoft 12", (1264, 1680), Palette16, 1.8),
|
||||||
'K2': ("Kindle 2", (600, 670), Palette15, 1.0),
|
'K11': ("Kindle 11", (1072, 1448), Palette16, 1.8),
|
||||||
'KDX': ("Kindle DX/DXG", (824, 1000), Palette16, 1.0),
|
'KPW5': ("Kindle Paperwhite 5/Signature Edition", (1236, 1648), Palette16, 1.8),
|
||||||
'K34': ("Kindle Keyboard/Touch", (600, 800), Palette16, 1.0),
|
'KS': ("Kindle Scribe", (1860, 2480), Palette16, 1.8),
|
||||||
'K57': ("Kindle 5/7", (600, 800), Palette16, 1.0),
|
|
||||||
'KPW': ("Kindle Paperwhite 1/2", (758, 1024), Palette16, 1.0),
|
|
||||||
'KV': ("Kindle Voyage", (1072, 1448), Palette16, 1.0),
|
|
||||||
'KPW34': ("Kindle Paperwhite 3/4/Oasis", (1072, 1448), Palette16, 1.0),
|
|
||||||
'K810': ("Kindle 8/10", (600, 800), Palette16, 1.0),
|
|
||||||
'KO': ("Kindle Oasis 2/3/Paperwhite 12", (1264, 1680), Palette16, 1.0),
|
|
||||||
'K11': ("Kindle 11", (1072, 1448), Palette16, 1.0),
|
|
||||||
'KPW5': ("Kindle Paperwhite 5/Signature Edition", (1236, 1648), Palette16, 1.0),
|
|
||||||
'KS1860': ("Kindle 1860", (1860, 1920), Palette16, 1.0),
|
|
||||||
'KS1920': ("Kindle 1920", (1920, 1920), Palette16, 1.0),
|
|
||||||
'KS1240': ("Kindle 1240", (1240, 1860), Palette16, 1.0),
|
|
||||||
'KS': ("Kindle Scribe 1/2", (1860, 2480), Palette16, 1.0),
|
|
||||||
'KCS': ("Kindle Colorsoft", (1264, 1680), Palette16, 1.0),
|
|
||||||
'KS3': ("Kindle Scribe 3", (1986, 2648), Palette16, 1.0),
|
|
||||||
'KSCS': ("Kindle Scribe Colorsoft", (1986, 2648), Palette16, 1.0),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ProfilesKindle = {
|
ProfilesKindle = {
|
||||||
@@ -116,35 +102,34 @@ class ProfileData:
|
|||||||
}
|
}
|
||||||
|
|
||||||
ProfilesKobo = {
|
ProfilesKobo = {
|
||||||
'KoMT': ("Kobo Mini/Touch", (600, 800), Palette16, 1.0),
|
'KoMT': ("Kobo Mini/Touch", (600, 800), Palette16, 1.8),
|
||||||
'KoG': ("Kobo Glo", (768, 1024), Palette16, 1.0),
|
'KoG': ("Kobo Glo", (768, 1024), Palette16, 1.8),
|
||||||
'KoGHD': ("Kobo Glo HD", (1072, 1448), Palette16, 1.0),
|
'KoGHD': ("Kobo Glo HD", (1072, 1448), Palette16, 1.8),
|
||||||
'KoA': ("Kobo Aura", (758, 1024), Palette16, 1.0),
|
'KoA': ("Kobo Aura", (758, 1024), Palette16, 1.8),
|
||||||
'KoAHD': ("Kobo Aura HD", (1080, 1440), Palette16, 1.0),
|
'KoAHD': ("Kobo Aura HD", (1080, 1440), Palette16, 1.8),
|
||||||
'KoAH2O': ("Kobo Aura H2O", (1080, 1430), Palette16, 1.0),
|
'KoAH2O': ("Kobo Aura H2O", (1080, 1430), Palette16, 1.8),
|
||||||
'KoAO': ("Kobo Aura ONE", (1404, 1872), Palette16, 1.0),
|
'KoAO': ("Kobo Aura ONE", (1404, 1872), Palette16, 1.8),
|
||||||
'KoN': ("Kobo Nia", (758, 1024), Palette16, 1.0),
|
'KoN': ("Kobo Nia", (758, 1024), Palette16, 1.8),
|
||||||
'KoC': ("Kobo Clara HD/Kobo Clara 2E", (1072, 1448), Palette16, 1.0),
|
'KoC': ("Kobo Clara HD/Kobo Clara 2E", (1072, 1448), Palette16, 1.8),
|
||||||
'KoCC': ("Kobo Clara Colour", (1072, 1448), Palette16, 1.0),
|
'KoCC': ("Kobo Clara Colour", (1072, 1448), Palette16, 1.8),
|
||||||
'KoL': ("Kobo Libra H2O/Kobo Libra 2", (1264, 1680), Palette16, 1.0),
|
'KoL': ("Kobo Libra H2O/Kobo Libra 2", (1264, 1680), Palette16, 1.8),
|
||||||
'KoLC': ("Kobo Libra Colour", (1264, 1680), Palette16, 1.0),
|
'KoLC': ("Kobo Libra Colour", (1264, 1680), Palette16, 1.8),
|
||||||
'KoF': ("Kobo Forma", (1440, 1920), Palette16, 1.0),
|
'KoF': ("Kobo Forma", (1440, 1920), Palette16, 1.8),
|
||||||
'KoS': ("Kobo Sage", (1440, 1920), Palette16, 1.0),
|
'KoS': ("Kobo Sage", (1440, 1920), Palette16, 1.8),
|
||||||
'KoE': ("Kobo Elipsa", (1404, 1872), Palette16, 1.0),
|
'KoE': ("Kobo Elipsa", (1404, 1872), Palette16, 1.8),
|
||||||
}
|
}
|
||||||
|
|
||||||
ProfilesRemarkable = {
|
ProfilesRemarkable = {
|
||||||
'Rmk1': ("reMarkable 1", (1404, 1872), Palette16, 1.0),
|
'Rmk1': ("reMarkable 1", (1404, 1872), Palette16, 1.8),
|
||||||
'Rmk2': ("reMarkable 2", (1404, 1872), Palette16, 1.0),
|
'Rmk2': ("reMarkable 2", (1404, 1872), Palette16, 1.8),
|
||||||
'RmkPP': ("reMarkable Paper Pro", (1620, 2160), Palette16, 1.0),
|
'RmkPP': ("reMarkable Paper Pro", (1620, 2160), Palette16, 1.8),
|
||||||
'RmkPPMove': ("reMarkable Paper Pro Move", (954, 1696), Palette16, 1.0),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Profiles = {
|
Profiles = {
|
||||||
**ProfilesKindle,
|
**ProfilesKindle,
|
||||||
**ProfilesKobo,
|
**ProfilesKobo,
|
||||||
**ProfilesRemarkable,
|
**ProfilesRemarkable,
|
||||||
'OTHER': ("Other", (0, 0), Palette16, 1.0),
|
'OTHER': ("Other", (0, 0), Palette16, 1.8),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -155,13 +140,8 @@ class ComicPageParser:
|
|||||||
self.source = source
|
self.source = source
|
||||||
self.size = self.opt.profileData[1]
|
self.size = self.opt.profileData[1]
|
||||||
self.payload = []
|
self.payload = []
|
||||||
|
self.image = Image.open(os.path.join(source[0], source[1])).convert('RGB')
|
||||||
# Detect corruption in source image, let caller catch any exceptions triggered.
|
self.color = self.colorCheck()
|
||||||
srcImgPath = os.path.join(source[0], source[1])
|
|
||||||
# Image.open(srcImgPath).verify()
|
|
||||||
with Image.open(srcImgPath) as im:
|
|
||||||
self.image = im.copy()
|
|
||||||
|
|
||||||
self.fill = self.fillCheck()
|
self.fill = self.fillCheck()
|
||||||
# backwards compatibility for Pillow >9.1.0
|
# backwards compatibility for Pillow >9.1.0
|
||||||
if not hasattr(Image, 'Resampling'):
|
if not hasattr(Image, 'Resampling'):
|
||||||
@@ -192,13 +172,13 @@ class ComicPageParser:
|
|||||||
new_image = Image.new("RGB", (int(width / 2), int(height*2)))
|
new_image = Image.new("RGB", (int(width / 2), int(height*2)))
|
||||||
new_image.paste(pageone, (0, 0))
|
new_image.paste(pageone, (0, 0))
|
||||||
new_image.paste(pagetwo, (0, height))
|
new_image.paste(pagetwo, (0, height))
|
||||||
self.payload.append(['N', self.source, new_image, self.fill])
|
self.payload.append(['N', self.source, new_image, self.color, self.fill])
|
||||||
elif (width > height) != (dstwidth > dstheight) and width <= dstheight and height <= dstwidth \
|
elif (width > height) != (dstwidth > dstheight) and width <= dstheight and height <= dstwidth \
|
||||||
and not self.opt.webtoon and self.opt.splitter == 1:
|
and not self.opt.webtoon and self.opt.splitter == 1:
|
||||||
spread = self.image
|
spread = self.image
|
||||||
if not self.opt.norotate:
|
if not self.opt.norotate:
|
||||||
spread = spread.rotate(90, Image.Resampling.BICUBIC, True)
|
spread = spread.rotate(90, Image.Resampling.BICUBIC, True)
|
||||||
self.payload.append(['R', self.source, spread, self.fill])
|
self.payload.append(['R', self.source, spread, self.color, self.fill])
|
||||||
elif (width > height) != (dstwidth > dstheight) and not self.opt.webtoon:
|
elif (width > height) != (dstwidth > dstheight) and not self.opt.webtoon:
|
||||||
if self.opt.splitter != 1:
|
if self.opt.splitter != 1:
|
||||||
if width > height:
|
if width > height:
|
||||||
@@ -213,15 +193,38 @@ class ComicPageParser:
|
|||||||
else:
|
else:
|
||||||
pageone = self.image.crop(leftbox)
|
pageone = self.image.crop(leftbox)
|
||||||
pagetwo = self.image.crop(rightbox)
|
pagetwo = self.image.crop(rightbox)
|
||||||
self.payload.append(['S1', self.source, pageone, self.fill])
|
self.payload.append(['S1', self.source, pageone, self.color, self.fill])
|
||||||
self.payload.append(['S2', self.source, pagetwo, self.fill])
|
self.payload.append(['S2', self.source, pagetwo, self.color, self.fill])
|
||||||
if self.opt.splitter > 0:
|
if self.opt.splitter > 0:
|
||||||
spread = self.image
|
spread = self.image
|
||||||
if not self.opt.norotate:
|
if not self.opt.norotate:
|
||||||
spread = spread.rotate(90, Image.Resampling.BICUBIC, True)
|
spread = spread.rotate(90, Image.Resampling.BICUBIC, True)
|
||||||
self.payload.append(['R', self.source, spread, self.fill])
|
self.payload.append(['R', self.source, spread,
|
||||||
|
self.color, self.fill])
|
||||||
else:
|
else:
|
||||||
self.payload.append(['N', self.source, self.image, self.fill])
|
self.payload.append(['N', self.source, self.image, self.color, self.fill])
|
||||||
|
|
||||||
|
def colorCheck(self):
|
||||||
|
if self.opt.webtoon:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
img = self.image.copy()
|
||||||
|
bands = img.getbands()
|
||||||
|
if bands == ('R', 'G', 'B') or bands == ('R', 'G', 'B', 'A'):
|
||||||
|
thumb = img.resize((40, 40))
|
||||||
|
SSE, bias = 0, [0, 0, 0]
|
||||||
|
bias = ImageStat.Stat(thumb).mean[:3]
|
||||||
|
bias = [b - sum(bias) / 3 for b in bias]
|
||||||
|
for pixel in thumb.getdata():
|
||||||
|
mu = sum(pixel) / 3
|
||||||
|
SSE += sum((pixel[i] - mu - bias[i]) * (pixel[i] - mu - bias[i]) for i in [0, 1, 2])
|
||||||
|
MSE = float(SSE) / (40 * 40)
|
||||||
|
if MSE > 22:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
def fillCheck(self):
|
def fillCheck(self):
|
||||||
if self.opt.bordersColor:
|
if self.opt.bordersColor:
|
||||||
@@ -263,263 +266,113 @@ class ComicPageParser:
|
|||||||
|
|
||||||
|
|
||||||
class ComicPage:
|
class ComicPage:
|
||||||
def __init__(self, options, mode, path, image, fill):
|
def __init__(self, options, mode, path, image, color, fill):
|
||||||
self.opt = options
|
self.opt = options
|
||||||
_, self.size, self.palette, self.gamma = self.opt.profileData
|
_, self.size, self.palette, self.gamma = self.opt.profileData
|
||||||
if self.opt.hq:
|
if self.opt.hq:
|
||||||
self.size = (int(self.size[0] * 1.5), int(self.size[1] * 1.5))
|
self.size = (int(self.size[0] * 1.5), int(self.size[1] * 1.5))
|
||||||
self.original_color_mode = image.mode
|
self.kindle_scribe_azw3 = (options.profile == 'KS') and (options.format in ('MOBI', 'EPUB'))
|
||||||
# TODO: color check earlier
|
self.image = image
|
||||||
self.image = image.convert("RGB")
|
self.color = color
|
||||||
self.fill = fill
|
self.fill = fill
|
||||||
self.rotated = False
|
self.rotated = False
|
||||||
self.orgPath = os.path.join(path[0], path[1])
|
self.orgPath = os.path.join(path[0], path[1])
|
||||||
self.targetPathStart = os.path.join(path[0], os.path.splitext(path[1])[0])
|
|
||||||
if 'N' in mode:
|
if 'N' in mode:
|
||||||
self.targetPathOrder = '-kcc-x'
|
self.targetPath = os.path.join(path[0], os.path.splitext(path[1])[0]) + '-KCC'
|
||||||
elif 'R' in mode:
|
elif 'R' in mode:
|
||||||
self.targetPathOrder = '-kcc-a' if options.rotatefirst else '-kcc-d'
|
self.targetPath = os.path.join(path[0], os.path.splitext(path[1])[0]) + '-KCC-A'
|
||||||
if not options.norotate:
|
self.rotated = True
|
||||||
self.rotated = True
|
|
||||||
elif 'S1' in mode:
|
elif 'S1' in mode:
|
||||||
self.targetPathOrder = '-kcc-b'
|
self.targetPath = os.path.join(path[0], os.path.splitext(path[1])[0]) + '-KCC-B'
|
||||||
elif 'S2' in mode:
|
elif 'S2' in mode:
|
||||||
self.targetPathOrder = '-kcc-c'
|
self.targetPath = os.path.join(path[0], os.path.splitext(path[1])[0]) + '-KCC-C'
|
||||||
# backwards compatibility for Pillow >9.1.0
|
# backwards compatibility for Pillow >9.1.0
|
||||||
if not hasattr(Image, 'Resampling'):
|
if not hasattr(Image, 'Resampling'):
|
||||||
Image.Resampling = Image
|
Image.Resampling = Image
|
||||||
|
|
||||||
@cached_property
|
|
||||||
def color(self):
|
|
||||||
if self.original_color_mode in ("L", "1"):
|
|
||||||
return False
|
|
||||||
if self.opt.webtoon:
|
|
||||||
return True
|
|
||||||
if self.calculate_color():
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
# cut off pixels from both ends of the histogram to remove jpg compression artifacts
|
|
||||||
# for better accuracy, you could split the image in half and analyze each half separately
|
|
||||||
def histograms_cutoff(self, cb_hist, cr_hist, cutoff=(2, 2)):
|
|
||||||
if cutoff == (0, 0):
|
|
||||||
return cb_hist, cr_hist
|
|
||||||
|
|
||||||
for h in cb_hist, cr_hist:
|
|
||||||
# get number of pixels
|
|
||||||
n = sum(h)
|
|
||||||
# remove cutoff% pixels from the low end
|
|
||||||
cut = int(n * cutoff[0] // 100)
|
|
||||||
for lo in range(256):
|
|
||||||
if cut > h[lo]:
|
|
||||||
cut = cut - h[lo]
|
|
||||||
h[lo] = 0
|
|
||||||
else:
|
|
||||||
h[lo] -= cut
|
|
||||||
cut = 0
|
|
||||||
if cut <= 0:
|
|
||||||
break
|
|
||||||
# remove cutoff% samples from the high end
|
|
||||||
cut = int(n * cutoff[1] // 100)
|
|
||||||
for hi in range(255, -1, -1):
|
|
||||||
if cut > h[hi]:
|
|
||||||
cut = cut - h[hi]
|
|
||||||
h[hi] = 0
|
|
||||||
else:
|
|
||||||
h[hi] -= cut
|
|
||||||
cut = 0
|
|
||||||
if cut <= 0:
|
|
||||||
break
|
|
||||||
return cb_hist, cr_hist
|
|
||||||
|
|
||||||
def color_precision(self, cb_hist_original, cr_hist_original, cutoff, diff_threshold):
|
|
||||||
cb_hist, cr_hist = self.histograms_cutoff(cb_hist_original.copy(), cr_hist_original.copy(), cutoff)
|
|
||||||
|
|
||||||
cb_nonzero = [i for i, e in enumerate(cb_hist) if e]
|
|
||||||
cr_nonzero = [i for i, e in enumerate(cr_hist) if e]
|
|
||||||
cb_spread = cb_nonzero[-1] - cb_nonzero[0]
|
|
||||||
cr_spread = cr_nonzero[-1] - cr_nonzero[0]
|
|
||||||
|
|
||||||
# bias adjustment, don't go lower than 7
|
|
||||||
SPREAD_THRESHOLD = 7
|
|
||||||
if self.opt.forcecolor:
|
|
||||||
if any([
|
|
||||||
cb_nonzero[0] > 128,
|
|
||||||
cr_nonzero[0] > 128,
|
|
||||||
cb_nonzero[-1] < 128,
|
|
||||||
cr_nonzero[-1] < 128,
|
|
||||||
]):
|
|
||||||
return True, True
|
|
||||||
elif cb_spread < SPREAD_THRESHOLD and cr_spread < SPREAD_THRESHOLD:
|
|
||||||
return True, False
|
|
||||||
|
|
||||||
DIFF_THRESHOLD = diff_threshold
|
|
||||||
if any([
|
|
||||||
cb_nonzero[0] <= 128 - DIFF_THRESHOLD,
|
|
||||||
cr_nonzero[0] <= 128 - DIFF_THRESHOLD,
|
|
||||||
cb_nonzero[-1] >= 128 + DIFF_THRESHOLD,
|
|
||||||
cr_nonzero[-1] >= 128 + DIFF_THRESHOLD,
|
|
||||||
]):
|
|
||||||
return True, True
|
|
||||||
|
|
||||||
return False, None
|
|
||||||
|
|
||||||
def calculate_color(self):
|
|
||||||
img = self.image.convert("YCbCr")
|
|
||||||
_, cb, cr = img.split()
|
|
||||||
cb_hist_original = cb.histogram()
|
|
||||||
cr_hist_original = cr.histogram()
|
|
||||||
|
|
||||||
# you can increase 22 but don't increase 10. 4 maybe can go higher
|
|
||||||
for cutoff, diff_threshold in [((0, 0), 22), ((.2, .2), 10), ((3, 3), 4)]:
|
|
||||||
done, decision = self.color_precision(cb_hist_original, cr_hist_original, cutoff, diff_threshold)
|
|
||||||
if done:
|
|
||||||
return decision
|
|
||||||
return False
|
|
||||||
|
|
||||||
def saveToDir(self):
|
def saveToDir(self):
|
||||||
try:
|
try:
|
||||||
flags = []
|
flags = []
|
||||||
|
if not self.opt.forcecolor and not self.opt.forcepng:
|
||||||
|
self.image = self.image.convert('L')
|
||||||
if self.rotated:
|
if self.rotated:
|
||||||
flags.append('Rotated')
|
flags.append('Rotated')
|
||||||
if self.fill != 'white':
|
if self.fill != 'white':
|
||||||
flags.append('BlackBackground')
|
flags.append('BlackBackground')
|
||||||
if self.opt.kindle_scribe_azw3 and self.image.size[1] > 1920:
|
if self.opt.forcepng:
|
||||||
w, h = self.image.size
|
self.image.info["transparency"] = None
|
||||||
targetPath = self.save_with_codec(self.image.crop((0, 0, w, 1920)), self.targetPathStart + self.targetPathOrder + '-above')
|
self.targetPath += '.png'
|
||||||
self.save_with_codec(self.image.crop((0, 1920, w, h)), self.targetPathStart + self.targetPathOrder + '-below')
|
self.image.save(self.targetPath, 'PNG', optimize=1)
|
||||||
elif self.opt.kindle_scribe_azw3:
|
|
||||||
targetPath = self.save_with_codec(self.image, self.targetPathStart + self.targetPathOrder + '-whole')
|
|
||||||
else:
|
else:
|
||||||
targetPath = self.save_with_codec(self.image, self.targetPathStart + self.targetPathOrder)
|
self.targetPath += '.jpg'
|
||||||
if os.path.isfile(self.orgPath):
|
if self.opt.mozjpeg:
|
||||||
os.remove(self.orgPath)
|
with io.BytesIO() as output:
|
||||||
return [Path(targetPath).name, flags]
|
self.image.save(output, format="JPEG", optimize=1, quality=85)
|
||||||
|
input_jpeg_bytes = output.getvalue()
|
||||||
|
output_jpeg_bytes = mozjpeg_lossless_optimization.optimize(input_jpeg_bytes)
|
||||||
|
with open(self.targetPath, "wb") as output_jpeg_file:
|
||||||
|
output_jpeg_file.write(output_jpeg_bytes)
|
||||||
|
else:
|
||||||
|
self.image.save(self.targetPath, 'JPEG', optimize=1, quality=85)
|
||||||
|
return [md5Checksum(self.targetPath), flags, self.orgPath]
|
||||||
except IOError as err:
|
except IOError as err:
|
||||||
raise RuntimeError('Cannot save image. ' + str(err))
|
raise RuntimeError('Cannot save image. ' + str(err))
|
||||||
|
|
||||||
def save_with_codec(self, image, targetPath):
|
def autocontrastImage(self):
|
||||||
if self.opt.forcepng:
|
|
||||||
image.info.pop('transparency', None)
|
|
||||||
if self.opt.iskindle and ('MOBI' in self.opt.format or 'EPUB' in self.opt.format):
|
|
||||||
targetPath += '.gif'
|
|
||||||
image.save(targetPath, 'GIF', optimize=1, interlace=False)
|
|
||||||
else:
|
|
||||||
targetPath += '.png'
|
|
||||||
image.save(targetPath, 'PNG', optimize=1)
|
|
||||||
else:
|
|
||||||
targetPath += '.jpg'
|
|
||||||
if self.opt.mozjpeg:
|
|
||||||
with io.BytesIO() as output:
|
|
||||||
image.save(output, format="JPEG", optimize=1, quality=self.opt.jpegquality)
|
|
||||||
input_jpeg_bytes = output.getvalue()
|
|
||||||
output_jpeg_bytes = mozjpeg_lossless_optimization.optimize(input_jpeg_bytes)
|
|
||||||
with open(targetPath, "wb") as output_jpeg_file:
|
|
||||||
output_jpeg_file.write(output_jpeg_bytes)
|
|
||||||
else:
|
|
||||||
image.save(targetPath, 'JPEG', optimize=1, quality=self.opt.jpegquality)
|
|
||||||
return targetPath
|
|
||||||
|
|
||||||
def gammaCorrectImage(self):
|
|
||||||
gamma = self.opt.gamma
|
gamma = self.opt.gamma
|
||||||
if gamma < 0.1:
|
if gamma < 0.1:
|
||||||
gamma = self.gamma
|
gamma = self.gamma
|
||||||
if self.gamma != 1.0 and self.color:
|
if self.gamma != 1.0 and self.color:
|
||||||
gamma = 1.0
|
gamma = 1.0
|
||||||
if gamma == 1.0:
|
if gamma == 1.0:
|
||||||
pass
|
self.image = ImageOps.autocontrast(self.image)
|
||||||
else:
|
else:
|
||||||
self.image = Image.eval(self.image, lambda a: int(255 * (a / 255.) ** gamma))
|
self.image = ImageOps.autocontrast(Image.eval(self.image, lambda a: int(255 * (a / 255.) ** gamma)))
|
||||||
|
|
||||||
def autocontrastImage(self):
|
|
||||||
if self.opt.webtoon:
|
|
||||||
return
|
|
||||||
if self.opt.noautocontrast:
|
|
||||||
return
|
|
||||||
if self.color and not self.opt.colorautocontrast:
|
|
||||||
return
|
|
||||||
|
|
||||||
# if image is extremely low contrast, that was probably intentional
|
|
||||||
extrema = self.image.convert('L').getextrema()
|
|
||||||
if extrema[1] - extrema[0] < (255 - 32 * 3):
|
|
||||||
return
|
|
||||||
|
|
||||||
if self.opt.autolevel:
|
|
||||||
self.autolevelImage()
|
|
||||||
|
|
||||||
self.image = ImageOps.autocontrast(self.image, preserve_tone=True)
|
|
||||||
|
|
||||||
def autolevelImage(self):
|
|
||||||
img = self.image
|
|
||||||
if self.color:
|
|
||||||
img = self.image.convert("YCbCr")
|
|
||||||
y, cb, cr = img.split()
|
|
||||||
img = y
|
|
||||||
else:
|
|
||||||
img = img.convert('L')
|
|
||||||
h = img.histogram()
|
|
||||||
most_common_dark_pixel_count = max(h[:64])
|
|
||||||
black_point = h.index(most_common_dark_pixel_count)
|
|
||||||
bp = black_point
|
|
||||||
img = img.point(lambda p: p if p > bp else bp)
|
|
||||||
if self.color:
|
|
||||||
self.image = Image.merge(mode='YCbCr', bands=[img, cb, cr]).convert('RGB')
|
|
||||||
else:
|
|
||||||
self.image = img
|
|
||||||
|
|
||||||
def convertToGrayscale(self):
|
|
||||||
self.image = self.image.convert('L')
|
|
||||||
|
|
||||||
def quantizeImage(self):
|
def quantizeImage(self):
|
||||||
# remove all color pixels from image, since colorCheck() has some tolerance
|
colors = len(self.palette) // 3
|
||||||
# quantize with a small number of color pixels in a mostly b/w image can have unexpected results
|
if colors < 256:
|
||||||
self.image = self.image.convert("RGB")
|
self.palette += self.palette[:3] * (256 - colors)
|
||||||
|
|
||||||
palImg = Image.new('P', (1, 1))
|
palImg = Image.new('P', (1, 1))
|
||||||
palImg.putpalette(self.palette)
|
palImg.putpalette(self.palette)
|
||||||
|
self.image = self.image.convert('L')
|
||||||
|
self.image = self.image.convert('RGB')
|
||||||
|
# Quantize is deprecated but new function call it internally anyway...
|
||||||
self.image = self.image.quantize(palette=palImg)
|
self.image = self.image.quantize(palette=palImg)
|
||||||
|
|
||||||
def optimizeForDisplay(self, eraserainbow, is_color):
|
|
||||||
# Erase rainbow artifacts for grayscale and color images by removing spectral frequencies that cause Moire interference with color filter array
|
|
||||||
if eraserainbow and all(dim > 1 for dim in self.image.size):
|
|
||||||
self.image = erase_rainbow_artifacts(self.image, is_color)
|
|
||||||
|
|
||||||
def resizeImage(self):
|
def resizeImage(self):
|
||||||
if self.opt.norotate and self.targetPathOrder in ('-kcc-a', '-kcc-d') and not self.opt.kindle_scribe_azw3:
|
# kindle scribe conversion to mobi is limited in resolution by kindlegen, same with send to kindle and epub
|
||||||
# TODO: Kindle Scribe case
|
if self.kindle_scribe_azw3:
|
||||||
if self.opt.kindle_azw3 and any(dim > 1920 for dim in self.image.size):
|
self.size = (1440, 1920)
|
||||||
self.image = ImageOps.contain(self.image, (1920, 1920), Image.Resampling.LANCZOS)
|
|
||||||
elif self.image.size[0] > self.size[0] * 2 or self.image.size[1] > self.size[1]:
|
|
||||||
self.image = ImageOps.contain(self.image, (self.size[0] * 2, self.size[1]), Image.Resampling.LANCZOS)
|
|
||||||
return
|
|
||||||
|
|
||||||
ratio_device = float(self.size[1]) / float(self.size[0])
|
ratio_device = float(self.size[1]) / float(self.size[0])
|
||||||
ratio_image = float(self.image.size[1]) / float(self.image.size[0])
|
ratio_image = float(self.image.size[1]) / float(self.image.size[0])
|
||||||
method = self.resize_method()
|
method = self.resize_method()
|
||||||
if self.opt.stretch:
|
if self.opt.stretch:
|
||||||
self.image = self.image.resize(self.size, method)
|
self.image = self.image.resize(self.size, method)
|
||||||
elif method == Image.Resampling.BICUBIC and not self.opt.upscale:
|
elif method == Image.Resampling.BICUBIC and not self.opt.upscale:
|
||||||
pass
|
if self.opt.format == 'CBZ' or self.opt.kfx:
|
||||||
|
borderw = int((self.size[0] - self.image.size[0]) / 2)
|
||||||
|
borderh = int((self.size[1] - self.image.size[1]) / 2)
|
||||||
|
self.image = ImageOps.expand(self.image, border=(borderw, borderh), fill=self.fill)
|
||||||
|
if self.image.size[0] != self.size[0] or self.image.size[1] != self.size[1]:
|
||||||
|
self.image = ImageOps.fit(self.image, self.size, method=method)
|
||||||
else: # if image bigger than device resolution or smaller with upscaling
|
else: # if image bigger than device resolution or smaller with upscaling
|
||||||
if abs(ratio_image - ratio_device) < AUTO_CROP_THRESHOLD:
|
if abs(ratio_image - ratio_device) < AUTO_CROP_THRESHOLD:
|
||||||
self.image = ImageOps.fit(self.image, self.size, method=method)
|
self.image = ImageOps.fit(self.image, self.size, method=method)
|
||||||
elif (self.opt.format in ('CBZ', 'PDF') or self.opt.kfx) and not self.opt.white_borders:
|
elif self.opt.format == 'CBZ' or self.opt.kfx:
|
||||||
self.image = ImageOps.pad(self.image, self.size, method=method, color=self.fill)
|
self.image = ImageOps.pad(self.image, self.size, method=method, color=self.fill)
|
||||||
else:
|
else:
|
||||||
|
if self.kindle_scribe_azw3:
|
||||||
|
self.size = (1860, 1920)
|
||||||
self.image = ImageOps.contain(self.image, self.size, method=method)
|
self.image = ImageOps.contain(self.image, self.size, method=method)
|
||||||
|
|
||||||
def resize_method(self):
|
def resize_method(self):
|
||||||
if self.image.size[0] < self.size[0] and self.image.size[1] < self.size[1]:
|
if self.image.size[0] <= self.size[0] and self.image.size[1] <= self.size[1]:
|
||||||
return Image.Resampling.BICUBIC
|
return Image.Resampling.BICUBIC
|
||||||
else:
|
else:
|
||||||
return Image.Resampling.LANCZOS
|
return Image.Resampling.LANCZOS
|
||||||
|
|
||||||
def maybeCrop(self, box, minimum):
|
def maybeCrop(self, box, minimum):
|
||||||
w, h = self.image.size
|
|
||||||
left, upper, right, lower = box
|
|
||||||
if self.opt.preservemargin:
|
|
||||||
ratio = 1 - self.opt.preservemargin / 100
|
|
||||||
box = left * ratio, upper * ratio, right + (w - right) * (1 - ratio), lower + (h - lower) * (1 - ratio)
|
|
||||||
box_area = (box[2] - box[0]) * (box[3] - box[1])
|
box_area = (box[2] - box[0]) * (box[3] - box[1])
|
||||||
image_area = self.image.size[0] * self.image.size[1]
|
image_area = self.image.size[0] * self.image.size[1]
|
||||||
if (box_area / image_area) >= minimum:
|
if (box_area / image_area) >= minimum:
|
||||||
@@ -529,29 +382,24 @@ class ComicPage:
|
|||||||
bbox = get_bbox_crop_margin_page_number(self.image, power, self.fill)
|
bbox = get_bbox_crop_margin_page_number(self.image, power, self.fill)
|
||||||
|
|
||||||
if bbox:
|
if bbox:
|
||||||
w, h = self.image.size
|
|
||||||
left, upper, right, lower = bbox
|
|
||||||
# don't crop more than 10% of image
|
|
||||||
bbox = (min(0.1*w, left), min(0.1*h, upper), max(0.9*w, right), max(0.9*h, lower))
|
|
||||||
self.maybeCrop(bbox, minimum)
|
self.maybeCrop(bbox, minimum)
|
||||||
|
|
||||||
def cropMargin(self, power, minimum):
|
def cropMargin(self, power, minimum):
|
||||||
bbox = get_bbox_crop_margin(self.image, power, self.fill)
|
bbox = get_bbox_crop_margin(self.image, power, self.fill)
|
||||||
|
|
||||||
if bbox:
|
if bbox:
|
||||||
w, h = self.image.size
|
|
||||||
left, upper, right, lower = bbox
|
|
||||||
# don't crop more than 10% of image
|
|
||||||
bbox = (min(0.1*w, left), min(0.1*h, upper), max(0.9*w, right), max(0.9*h, lower))
|
|
||||||
self.maybeCrop(bbox, minimum)
|
self.maybeCrop(bbox, minimum)
|
||||||
|
|
||||||
def cropInterPanelEmptySections(self, direction):
|
|
||||||
self.image = crop_empty_inter_panel(self.image, direction, background_color=self.fill)
|
|
||||||
|
|
||||||
class Cover:
|
class Cover:
|
||||||
def __init__(self, source, opt):
|
def __init__(self, source, target, opt, tomeid):
|
||||||
self.options = opt
|
self.options = opt
|
||||||
self.source = source
|
self.source = source
|
||||||
|
self.target = target
|
||||||
|
if tomeid == 0:
|
||||||
|
self.tomeid = 1
|
||||||
|
else:
|
||||||
|
self.tomeid = tomeid
|
||||||
self.image = Image.open(source)
|
self.image = Image.open(source)
|
||||||
# backwards compatibility for Pillow >9.1.0
|
# backwards compatibility for Pillow >9.1.0
|
||||||
if not hasattr(Image, 'Resampling'):
|
if not hasattr(Image, 'Resampling'):
|
||||||
@@ -560,59 +408,22 @@ class Cover:
|
|||||||
|
|
||||||
def process(self):
|
def process(self):
|
||||||
self.image = self.image.convert('RGB')
|
self.image = self.image.convert('RGB')
|
||||||
self.image = ImageOps.autocontrast(self.image, preserve_tone=True)
|
self.image = ImageOps.autocontrast(self.image)
|
||||||
if not self.options.forcecolor:
|
if not self.options.forcecolor:
|
||||||
self.image = self.image.convert('L')
|
self.image = self.image.convert('L')
|
||||||
self.crop_main_cover()
|
self.image.thumbnail(self.options.profileData[1], Image.Resampling.LANCZOS)
|
||||||
|
self.save()
|
||||||
|
|
||||||
size = list(self.options.profileData[1])
|
def save(self):
|
||||||
if self.options.kindle_scribe_azw3:
|
|
||||||
size[0] = min(size[0], 1920)
|
|
||||||
size[1] = min(size[1], 1920)
|
|
||||||
if self.options.coverfill and not self.options.kindle_scribe_azw3:
|
|
||||||
# TODO: Kindle Scribe case
|
|
||||||
self.image = ImageOps.fit(self.image, tuple(size), Image.Resampling.LANCZOS, centering=(0.5, 0.5))
|
|
||||||
else:
|
|
||||||
self.image.thumbnail(tuple(size), Image.Resampling.LANCZOS)
|
|
||||||
|
|
||||||
def crop_main_cover(self):
|
|
||||||
w, h = self.image.size
|
|
||||||
if w / h > 2:
|
|
||||||
if self.options.righttoleft:
|
|
||||||
self.image = self.image.crop((w/6, 0, w/2 - w * 0.02, h))
|
|
||||||
else:
|
|
||||||
self.image = self.image.crop((w/2 + w * 0.02, 0, 5/6 * w, h))
|
|
||||||
elif w / h > 1.34:
|
|
||||||
if self.options.righttoleft:
|
|
||||||
self.image = self.image.crop((0, 0, w/2 - w * 0.03, h))
|
|
||||||
else:
|
|
||||||
self.image = self.image.crop((w/2 + w * 0.03, 0, w, h))
|
|
||||||
|
|
||||||
def save_to_epub(self, target, tomeid, len_tomes=0):
|
|
||||||
try:
|
try:
|
||||||
if tomeid == 0:
|
self.image.save(self.target, "JPEG", optimize=1, quality=85)
|
||||||
self.image.save(target, "JPEG", optimize=1, quality=self.options.jpegquality)
|
|
||||||
else:
|
|
||||||
copy = self.image.copy()
|
|
||||||
draw = ImageDraw.Draw(copy)
|
|
||||||
w, h = copy.size
|
|
||||||
draw.text(
|
|
||||||
xy=(w/2, h * .85),
|
|
||||||
text=f'{tomeid}/{len_tomes}',
|
|
||||||
anchor='ms',
|
|
||||||
font_size=h//7,
|
|
||||||
fill=255,
|
|
||||||
stroke_fill=0,
|
|
||||||
stroke_width=25
|
|
||||||
)
|
|
||||||
copy.save(target, "JPEG", optimize=1, quality=self.options.jpegquality)
|
|
||||||
except IOError:
|
except IOError:
|
||||||
raise RuntimeError('Failed to save cover.')
|
raise RuntimeError('Failed to save cover.')
|
||||||
|
|
||||||
def saveToKindle(self, kindle, asin):
|
def saveToKindle(self, kindle, asin):
|
||||||
self.image = ImageOps.contain(self.image, (300, 470), Image.Resampling.LANCZOS)
|
self.image = self.image.resize((300, 470), Image.Resampling.LANCZOS)
|
||||||
try:
|
try:
|
||||||
self.image.save(os.path.join(kindle.path.split('documents')[0], 'system', 'thumbnails',
|
self.image.save(os.path.join(kindle.path.split('documents')[0], 'system', 'thumbnails',
|
||||||
'thumbnail_' + asin + '_EBOK_portrait.jpg'), 'JPEG', optimize=1, quality=self.options.jpegquality)
|
'thumbnail_' + asin + '_EBOK_portrait.jpg'), 'JPEG', optimize=1, quality=85)
|
||||||
except IOError:
|
except IOError:
|
||||||
raise RuntimeError('Failed to upload cover.')
|
raise RuntimeError('Failed to upload cover.')
|
||||||
|
|||||||
@@ -1,78 +0,0 @@
|
|||||||
from PIL import Image, ImageFilter, ImageOps, ImageFile
|
|
||||||
import numpy as np
|
|
||||||
from typing import Literal
|
|
||||||
from .common_crop import threshold_from_power, group_close_values
|
|
||||||
|
|
||||||
ImageFile.LOAD_TRUNCATED_IMAGES = True
|
|
||||||
|
|
||||||
|
|
||||||
'''
|
|
||||||
Crops inter-panel empty spaces (ignores empty spaces near borders - for that use crop margins).
|
|
||||||
|
|
||||||
Parameters:
|
|
||||||
img (PIL image): A PIL image.
|
|
||||||
direction (horizontal or vertical or both): To crop rows (horizontal), cols (vertical) or both.
|
|
||||||
keep (float): Distance to keep between panels after cropping (in percentage relative to the original distance).
|
|
||||||
background_color (string): 'white' for white background, anything else for black.
|
|
||||||
Returns:
|
|
||||||
img (PIL image): A PIL image after cropping empty sections.
|
|
||||||
'''
|
|
||||||
def crop_empty_inter_panel(img, direction: Literal["horizontal", "vertical", "both"], keep=0.04, background_color='white'):
|
|
||||||
img_temp = img
|
|
||||||
|
|
||||||
if img.mode != 'L':
|
|
||||||
img_temp = ImageOps.grayscale(img_temp)
|
|
||||||
|
|
||||||
if background_color != 'white':
|
|
||||||
img_temp = ImageOps.invert(img_temp)
|
|
||||||
|
|
||||||
img_mat = np.array(img)
|
|
||||||
|
|
||||||
power = 1
|
|
||||||
img_temp = ImageOps.autocontrast(img_temp, 1).filter(ImageFilter.BoxBlur(1))
|
|
||||||
img_temp = img_temp.point(lambda p: 255 if p <= threshold_from_power(power) else 0)
|
|
||||||
|
|
||||||
if direction in ["horizontal", "both"]:
|
|
||||||
rows_idx_to_remove = empty_sections(img_temp, keep, horizontal=True)
|
|
||||||
img_mat = np.delete(img_mat, rows_idx_to_remove, 0)
|
|
||||||
|
|
||||||
if direction in ["vertical", "both"]:
|
|
||||||
cols_idx_to_remove = empty_sections(img_temp, keep, horizontal=False)
|
|
||||||
img_mat = np.delete(img_mat, cols_idx_to_remove, 1)
|
|
||||||
|
|
||||||
return Image.fromarray(img_mat)
|
|
||||||
|
|
||||||
|
|
||||||
'''
|
|
||||||
Finds empty sections (excluding near borders).
|
|
||||||
|
|
||||||
Parameters:
|
|
||||||
img (PIL image): A PIL image.
|
|
||||||
keep (float): Distance to keep between panels after cropping (in percentage relative to the original distance).
|
|
||||||
horizontal (boolean): True to find empty rows, False to find empty columns.
|
|
||||||
Returns:
|
|
||||||
Itertable (list or NumPy array): indices of rows or columns to remove.
|
|
||||||
'''
|
|
||||||
def empty_sections(img, keep, horizontal=True):
|
|
||||||
axis = 1 if horizontal else 0
|
|
||||||
|
|
||||||
img_mat = np.array(img)
|
|
||||||
img_mat_max = np.max(img_mat, axis=axis)
|
|
||||||
img_mat_empty_idx = np.where(img_mat_max == 0)[0]
|
|
||||||
|
|
||||||
empty_sections = group_close_values(img_mat_empty_idx, 1)
|
|
||||||
sections_to_remove = []
|
|
||||||
for section in empty_sections:
|
|
||||||
if section[1] < img.size[1] * 0.99 and section[0] > img.size[1] * 0.01: # if not near borders
|
|
||||||
sections_to_remove.append(section)
|
|
||||||
|
|
||||||
if len(sections_to_remove) != 0:
|
|
||||||
sections_to_remove_after_keep = [(int(x1+(keep/2)*(x2-x1)), int(x2-(keep/2)*(x2-x1))) for x1,x2 in sections_to_remove]
|
|
||||||
idx_to_remove = np.concatenate([np.arange(x1, x2) for x1,x2 in sections_to_remove_after_keep])
|
|
||||||
|
|
||||||
return idx_to_remove
|
|
||||||
|
|
||||||
return []
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -20,7 +20,6 @@ import os
|
|||||||
from xml.dom.minidom import parse, Document
|
from xml.dom.minidom import parse, Document
|
||||||
from tempfile import mkdtemp
|
from tempfile import mkdtemp
|
||||||
from shutil import rmtree
|
from shutil import rmtree
|
||||||
from xml.sax.saxutils import unescape
|
|
||||||
from . import comicarchive
|
from . import comicarchive
|
||||||
|
|
||||||
|
|
||||||
@@ -53,19 +52,19 @@ class MetadataParser:
|
|||||||
|
|
||||||
def parseXML(self):
|
def parseXML(self):
|
||||||
if len(self.rawdata.getElementsByTagName('Series')) != 0:
|
if len(self.rawdata.getElementsByTagName('Series')) != 0:
|
||||||
self.data['Series'] = unescape(self.rawdata.getElementsByTagName('Series')[0].firstChild.nodeValue)
|
self.data['Series'] = self.rawdata.getElementsByTagName('Series')[0].firstChild.nodeValue
|
||||||
if len(self.rawdata.getElementsByTagName('Volume')) != 0:
|
if len(self.rawdata.getElementsByTagName('Volume')) != 0:
|
||||||
self.data['Volume'] = self.rawdata.getElementsByTagName('Volume')[0].firstChild.nodeValue
|
self.data['Volume'] = self.rawdata.getElementsByTagName('Volume')[0].firstChild.nodeValue
|
||||||
if len(self.rawdata.getElementsByTagName('Number')) != 0:
|
if len(self.rawdata.getElementsByTagName('Number')) != 0:
|
||||||
self.data['Number'] = self.rawdata.getElementsByTagName('Number')[0].firstChild.nodeValue
|
self.data['Number'] = self.rawdata.getElementsByTagName('Number')[0].firstChild.nodeValue
|
||||||
if len(self.rawdata.getElementsByTagName('Summary')) != 0:
|
if len(self.rawdata.getElementsByTagName('Summary')) != 0:
|
||||||
self.data['Summary'] = unescape(self.rawdata.getElementsByTagName('Summary')[0].firstChild.nodeValue)
|
self.data['Summary'] = self.rawdata.getElementsByTagName('Summary')[0].firstChild.nodeValue
|
||||||
if len(self.rawdata.getElementsByTagName('Title')) != 0:
|
if len(self.rawdata.getElementsByTagName('Title')) != 0:
|
||||||
self.data['Title'] = unescape(self.rawdata.getElementsByTagName('Title')[0].firstChild.nodeValue)
|
self.data['Title'] = self.rawdata.getElementsByTagName('Title')[0].firstChild.nodeValue
|
||||||
for field in ['Writer', 'Penciller', 'Inker', 'Colorist']:
|
for field in ['Writer', 'Penciller', 'Inker', 'Colorist']:
|
||||||
if len(self.rawdata.getElementsByTagName(field)) != 0:
|
if len(self.rawdata.getElementsByTagName(field)) != 0:
|
||||||
for person in self.rawdata.getElementsByTagName(field)[0].firstChild.nodeValue.split(', '):
|
for person in self.rawdata.getElementsByTagName(field)[0].firstChild.nodeValue.split(', '):
|
||||||
self.data[field + 's'].append(unescape(person))
|
self.data[field + 's'].append(person)
|
||||||
self.data[field + 's'] = list(set(self.data[field + 's']))
|
self.data[field + 's'] = list(set(self.data[field + 's']))
|
||||||
self.data[field + 's'].sort()
|
self.data[field + 's'].sort()
|
||||||
if len(self.rawdata.getElementsByTagName('Page')) != 0:
|
if len(self.rawdata.getElementsByTagName('Page')) != 0:
|
||||||
@@ -123,4 +122,4 @@ class MetadataParser:
|
|||||||
cbx.addFile(tmpXML)
|
cbx.addFile(tmpXML)
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
raise UserWarning(e)
|
raise UserWarning(e)
|
||||||
rmtree(workdir, True)
|
rmtree(workdir)
|
||||||
|
|||||||
@@ -1,9 +1,5 @@
|
|||||||
from PIL import ImageOps, ImageFilter, ImageFile
|
from PIL import ImageOps, ImageFilter
|
||||||
import numpy as np
|
import numpy as np
|
||||||
from .common_crop import threshold_from_power, group_close_values
|
|
||||||
|
|
||||||
ImageFile.LOAD_TRUNCATED_IMAGES = True
|
|
||||||
|
|
||||||
|
|
||||||
'''
|
'''
|
||||||
Some assupmptions on the page number sizes
|
Some assupmptions on the page number sizes
|
||||||
@@ -54,13 +50,13 @@ def get_bbox_crop_margin_page_number(img, power=1, background_color='white'):
|
|||||||
'''
|
'''
|
||||||
threshold = threshold_from_power(power)
|
threshold = threshold_from_power(power)
|
||||||
bw_img = img.point(lambda p: 255 if p <= threshold else 0)
|
bw_img = img.point(lambda p: 255 if p <= threshold else 0)
|
||||||
ignore_pixels_near_edge(bw_img)
|
|
||||||
bw_bbox = bw_img.getbbox()
|
bw_bbox = bw_img.getbbox()
|
||||||
|
|
||||||
if not bw_bbox: # bbox cannot be found in case that the entire resulted image is black.
|
if not bw_bbox: # bbox cannot be found in case that the entire resulted image is black.
|
||||||
return None
|
return None
|
||||||
|
|
||||||
left, top_y_pos, right, bot_y_pos = bw_bbox
|
left, top_y_pos, right, bot_y_pos = bw_bbox
|
||||||
|
|
||||||
'''
|
'''
|
||||||
We inspect the lower bottom part of the image where we suspect might be a page number.
|
We inspect the lower bottom part of the image where we suspect might be a page number.
|
||||||
We assume that page number consist of 1 to 3 digits and the total min and max size of the number
|
We assume that page number consist of 1 to 3 digits and the total min and max size of the number
|
||||||
@@ -77,7 +73,7 @@ def get_bbox_crop_margin_page_number(img, power=1, background_color='white'):
|
|||||||
img_part_mat = np.array(img_part)
|
img_part_mat = np.array(img_part)
|
||||||
window_groups = []
|
window_groups = []
|
||||||
for i in range(img_part.size[1]):
|
for i in range(img_part.size[1]):
|
||||||
row_groups = [(g[0], g[1], i, i) for g in group_close_values(np.where(img_part_mat[i] <= threshold)[0], img.size[0]*max_dist_size[0])]
|
row_groups = [(g[0], g[1], i, i) for g in group_pixels(img_part_mat[i], img.size[0]*max_dist_size[0], threshold)]
|
||||||
window_groups.extend(row_groups)
|
window_groups.extend(row_groups)
|
||||||
|
|
||||||
window_groups = np.array(window_groups)
|
window_groups = np.array(window_groups)
|
||||||
@@ -113,6 +109,7 @@ def get_bbox_crop_margin_page_number(img, power=1, background_color='white'):
|
|||||||
cropped_bbox = (0, 0, img.size[0], bot_y_pos-(window_h-boxes_in_same_y_range[0][2]+1))
|
cropped_bbox = (0, 0, img.size[0], bot_y_pos-(window_h-boxes_in_same_y_range[0][2]+1))
|
||||||
|
|
||||||
cropped_bbox = bw_img.crop(cropped_bbox).getbbox()
|
cropped_bbox = bw_img.crop(cropped_bbox).getbbox()
|
||||||
|
|
||||||
return cropped_bbox
|
return cropped_bbox
|
||||||
|
|
||||||
|
|
||||||
@@ -144,27 +141,35 @@ def get_bbox_crop_margin(img, power=1, background_color='white'):
|
|||||||
'''
|
'''
|
||||||
threshold = threshold_from_power(power)
|
threshold = threshold_from_power(power)
|
||||||
bw_img = img.point(lambda p: 255 if p <= threshold else 0)
|
bw_img = img.point(lambda p: 255 if p <= threshold else 0)
|
||||||
|
|
||||||
ignore_pixels_near_edge(bw_img)
|
|
||||||
|
|
||||||
return bw_img.getbbox()
|
return bw_img.getbbox()
|
||||||
|
|
||||||
def ignore_pixels_near_edge(bw_img):
|
|
||||||
w, h = bw_img.size
|
'''
|
||||||
edge_bbox = [
|
Groups close pixels together (x axis)
|
||||||
(0, 0, w, int(0.02 * h)),
|
'''
|
||||||
(0, int(0.98 * h), w, h),
|
def group_pixels(row, max_dist_tolerated, threshold):
|
||||||
(0, 0, int(0.02 * w), h),
|
groups = []
|
||||||
(int(0.98 * w), 0, w, h)
|
idx = np.where(row <= threshold)[0]
|
||||||
]
|
|
||||||
for box in edge_bbox:
|
group_start = -1
|
||||||
edge = bw_img.crop(box)
|
group_end = 0
|
||||||
h = edge.histogram()
|
for i in range(len(idx)):
|
||||||
if not edge.height or not edge.width:
|
dist = idx[i] - group_end
|
||||||
continue
|
if group_start == -1:
|
||||||
imperfections = h[255] / (edge.height * edge.width)
|
group_start = idx[i]
|
||||||
if imperfections > 0 and imperfections < .02:
|
group_end = idx[i]
|
||||||
bw_img.paste(im=0, box=box)
|
elif dist <= max_dist_tolerated:
|
||||||
|
group_end = idx[i]
|
||||||
|
else:
|
||||||
|
groups.append((group_start, group_end))
|
||||||
|
group_start = -1
|
||||||
|
group_end = -1
|
||||||
|
|
||||||
|
if group_start != -1:
|
||||||
|
groups.append((group_start, group_end))
|
||||||
|
|
||||||
|
return groups
|
||||||
|
|
||||||
|
|
||||||
def box_intersect(box1, box2, max_dist):
|
def box_intersect(box1, box2, max_dist):
|
||||||
@@ -204,3 +209,7 @@ def merge_boxes(boxes, max_dist_tolerated):
|
|||||||
else:
|
else:
|
||||||
j += 1
|
j += 1
|
||||||
return boxes
|
return boxes
|
||||||
|
|
||||||
|
|
||||||
|
def threshold_from_power(power):
|
||||||
|
return 240-(power*64)
|
||||||
@@ -22,6 +22,8 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
import os
|
import os
|
||||||
|
from random import choice
|
||||||
|
from string import ascii_uppercase, digits
|
||||||
|
|
||||||
# skip stray images a few pixels in size in some PDFs
|
# skip stray images a few pixels in size in some PDFs
|
||||||
# typical images are many thousands in length
|
# typical images are many thousands in length
|
||||||
@@ -30,9 +32,10 @@ STRAY_IMAGE_LENGTH_THRESHOLD = 300
|
|||||||
|
|
||||||
|
|
||||||
class PdfJpgExtract:
|
class PdfJpgExtract:
|
||||||
def __init__(self, fname, fullPath):
|
def __init__(self, fname):
|
||||||
self.fname = fname
|
self.fname = fname
|
||||||
self.path = fullPath
|
self.filename = os.path.splitext(fname)
|
||||||
|
self.path = self.filename[0] + "-KCC-" + ''.join(choice(ascii_uppercase + digits) for _ in range(3))
|
||||||
|
|
||||||
def getPath(self):
|
def getPath(self):
|
||||||
return self.path
|
return self.path
|
||||||
@@ -45,6 +48,7 @@ class PdfJpgExtract:
|
|||||||
endfix = 2
|
endfix = 2
|
||||||
i = 0
|
i = 0
|
||||||
njpg = 0
|
njpg = 0
|
||||||
|
os.makedirs(self.path)
|
||||||
while True:
|
while True:
|
||||||
istream = pdf.find(b"stream", i)
|
istream = pdf.find(b"stream", i)
|
||||||
if istream < 0:
|
if istream < 0:
|
||||||
@@ -67,9 +71,9 @@ class PdfJpgExtract:
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
jpg = pdf[istart:iend]
|
jpg = pdf[istart:iend]
|
||||||
jpgfile = open(os.path.join(self.path, "jpg%d.jpg" % njpg), "wb")
|
jpgfile = open(self.path + "/jpg%d.jpg" % njpg, "wb")
|
||||||
jpgfile.write(jpg)
|
jpgfile.write(jpg)
|
||||||
jpgfile.close()
|
jpgfile.close()
|
||||||
njpg += 1
|
njpg += 1
|
||||||
|
|
||||||
return njpg
|
return self.path, njpg
|
||||||
|
|||||||
@@ -1,249 +0,0 @@
|
|||||||
import numpy as np
|
|
||||||
from PIL import Image, ImageFile
|
|
||||||
|
|
||||||
ImageFile.LOAD_TRUNCATED_IMAGES = True
|
|
||||||
|
|
||||||
|
|
||||||
def fourier_transform_image(img):
|
|
||||||
"""
|
|
||||||
Memory-optimized version that modifies the array in place when possible.
|
|
||||||
"""
|
|
||||||
# Convert with minimal copy
|
|
||||||
img_array = np.asarray(img, dtype=np.float32)
|
|
||||||
|
|
||||||
# Use rfft2 if the image is real to save memory
|
|
||||||
# and computation time (approximately 2x faster)
|
|
||||||
fft_result = np.fft.rfft2(img_array)
|
|
||||||
|
|
||||||
return fft_result
|
|
||||||
|
|
||||||
def attenuate_diagonal_frequencies(fft_spectrum, freq_threshold=0.30, target_angle=135,
|
|
||||||
angle_tolerance=10, attenuation_factor=0.10):
|
|
||||||
"""
|
|
||||||
Attenuates specific frequencies in the Fourier domain (optimized version for rfft2).
|
|
||||||
|
|
||||||
Args:
|
|
||||||
fft_spectrum: Result of 2D real Fourier transform (from rfft2)
|
|
||||||
freq_threshold: Frequency threshold in cycles/pixel (default: 0.3, theoretical max: 0.5)
|
|
||||||
target_angle: Target angle in degrees (default: 135)
|
|
||||||
angle_tolerance: Angular tolerance in degrees (default: 15)
|
|
||||||
attenuation_factor: Attenuation factor (0.1 = 90% attenuation)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
np.ndarray: Modified FFT with applied attenuation (same format as input)
|
|
||||||
"""
|
|
||||||
|
|
||||||
# Get dimensions of the rfft2 result
|
|
||||||
if fft_spectrum.ndim == 2:
|
|
||||||
height, width_rfft = fft_spectrum.shape
|
|
||||||
else: # 3D array (color channels)
|
|
||||||
height, width_rfft = fft_spectrum.shape[:2]
|
|
||||||
|
|
||||||
# For rfft2, the original width is (width_rfft - 1) * 2
|
|
||||||
width_original = (width_rfft - 1) * 2
|
|
||||||
|
|
||||||
# Create frequency grids for rfft2 format
|
|
||||||
freq_y = np.fft.fftfreq(height, d=1.0)
|
|
||||||
freq_x = np.fft.rfftfreq(width_original, d=1.0) # Use rfftfreq for the X dimension
|
|
||||||
|
|
||||||
|
|
||||||
# Use broadcasting to create grids without meshgrid (more efficient)
|
|
||||||
freq_y_grid = freq_y.reshape(-1, 1) # Column
|
|
||||||
freq_x_grid = freq_x.reshape(1, -1) # Row
|
|
||||||
|
|
||||||
# Calculate squared radial frequencies (avoid sqrt)
|
|
||||||
freq_radial_sq = freq_x_grid**2 + freq_y_grid**2
|
|
||||||
freq_threshold_sq = freq_threshold**2
|
|
||||||
|
|
||||||
# Frequency condition
|
|
||||||
freq_condition = freq_radial_sq >= freq_threshold_sq
|
|
||||||
|
|
||||||
# Early exit if no frequency satisfies the condition
|
|
||||||
if not np.any(freq_condition):
|
|
||||||
return fft_spectrum
|
|
||||||
|
|
||||||
# Calculate angles only where necessary
|
|
||||||
# Use atan2 directly with broadcasting
|
|
||||||
angles_rad = np.arctan2(freq_y_grid, freq_x_grid)
|
|
||||||
|
|
||||||
# Convert to degrees and normalize in a single operation
|
|
||||||
angles_deg = np.rad2deg(angles_rad) % 360
|
|
||||||
|
|
||||||
# Calculation of complementary angle
|
|
||||||
target_angle_2 = (target_angle + 180) % 360
|
|
||||||
|
|
||||||
# Calulation of perpendicular angles (135° + 45° to maximize compatibility until we know for sure which angle configure for each device)
|
|
||||||
target_angle_3 = (target_angle + 90) % 360
|
|
||||||
target_angle_4 = (target_angle_3 + 180) % 360
|
|
||||||
|
|
||||||
# Create angular conditions in a vectorized way
|
|
||||||
angle_condition = np.zeros_like(angles_deg, dtype=bool)
|
|
||||||
|
|
||||||
# Process both angles simultaneously
|
|
||||||
for angle in [target_angle, target_angle_2, target_angle_3, target_angle_4]:
|
|
||||||
min_angle = (angle - angle_tolerance) % 360
|
|
||||||
max_angle = (angle + angle_tolerance) % 360
|
|
||||||
|
|
||||||
if min_angle > max_angle: # Interval crosses 0°
|
|
||||||
angle_condition |= (angles_deg >= min_angle) | (angles_deg <= max_angle)
|
|
||||||
else: # Normal interval
|
|
||||||
angle_condition |= (angles_deg >= min_angle) & (angles_deg <= max_angle)
|
|
||||||
|
|
||||||
# Combine conditions
|
|
||||||
combined_condition = freq_condition & angle_condition
|
|
||||||
|
|
||||||
# Apply attenuation directly (avoid creating a full mask)
|
|
||||||
if attenuation_factor == 0:
|
|
||||||
# Special case: complete suppression
|
|
||||||
if fft_spectrum.ndim == 2:
|
|
||||||
fft_spectrum[combined_condition] = 0
|
|
||||||
else: # 3D array
|
|
||||||
fft_spectrum[combined_condition, :] = 0
|
|
||||||
return fft_spectrum
|
|
||||||
elif attenuation_factor == 1:
|
|
||||||
# Special case: no attenuation
|
|
||||||
return fft_spectrum
|
|
||||||
else:
|
|
||||||
# General case: partial attenuation
|
|
||||||
if fft_spectrum.ndim == 2:
|
|
||||||
fft_spectrum[combined_condition] *= attenuation_factor
|
|
||||||
else: # 3D array
|
|
||||||
fft_spectrum[combined_condition, :] *= attenuation_factor
|
|
||||||
return fft_spectrum
|
|
||||||
|
|
||||||
def inverse_fourier_transform_image(fft_spectrum, is_color, original_shape=None):
|
|
||||||
"""
|
|
||||||
Performs an optimized inverse Fourier transform to reconstruct a PIL image.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
fft_spectrum: Fourier transform result (complex array from rfft2)
|
|
||||||
is_color: Boolean indicating if the image is to be treated as color
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
PIL.Image: Reconstructed image
|
|
||||||
"""
|
|
||||||
# Perform inverse Fourier transform with original shape if provided
|
|
||||||
if original_shape is not None:
|
|
||||||
img_reconstructed = np.fft.irfft2(fft_spectrum, s=original_shape)
|
|
||||||
else:
|
|
||||||
img_reconstructed = np.fft.irfft2(fft_spectrum)
|
|
||||||
|
|
||||||
# Normalize values between 0 and 255
|
|
||||||
img_reconstructed = np.clip(img_reconstructed, 0, 255)
|
|
||||||
img_reconstructed = img_reconstructed.astype(np.uint8)
|
|
||||||
|
|
||||||
# Convert to PIL image
|
|
||||||
if is_color and img_reconstructed.ndim == 3:
|
|
||||||
pil_image = Image.fromarray(img_reconstructed, mode='RGB')
|
|
||||||
else:
|
|
||||||
pil_image = Image.fromarray(img_reconstructed, mode='L')
|
|
||||||
|
|
||||||
return pil_image
|
|
||||||
|
|
||||||
def rgb_to_yuv(rgb_array):
|
|
||||||
"""
|
|
||||||
Convert RGB to YUV color space.
|
|
||||||
Y = luminance, U and V = chrominance
|
|
||||||
"""
|
|
||||||
# Coefficients for RGB to YUV conversion
|
|
||||||
rgb_to_yuv_matrix = np.array([
|
|
||||||
[0.299, 0.587, 0.114], # Y
|
|
||||||
[-0.14713, -0.28886, 0.436], # U
|
|
||||||
[0.615, -0.51499, -0.10001] # V
|
|
||||||
])
|
|
||||||
|
|
||||||
# Reshape for matrix multiplication
|
|
||||||
original_shape = rgb_array.shape
|
|
||||||
rgb_flat = rgb_array.reshape(-1, 3)
|
|
||||||
|
|
||||||
# Apply transformation
|
|
||||||
yuv_flat = rgb_flat @ rgb_to_yuv_matrix.T
|
|
||||||
|
|
||||||
# Reshape back
|
|
||||||
yuv_array = yuv_flat.reshape(original_shape)
|
|
||||||
|
|
||||||
return yuv_array
|
|
||||||
|
|
||||||
def yuv_to_rgb(yuv_array):
|
|
||||||
"""
|
|
||||||
Convert YUV to RGB color space.
|
|
||||||
"""
|
|
||||||
# Coefficients for YUV to RGB conversion
|
|
||||||
yuv_to_rgb_matrix = np.array([
|
|
||||||
[1.0, 0.0, 1.13983], # R
|
|
||||||
[1.0, -0.39465, -0.58060], # G
|
|
||||||
[1.0, 2.03211, 0.0] # B
|
|
||||||
])
|
|
||||||
|
|
||||||
# Reshape for matrix multiplication
|
|
||||||
original_shape = yuv_array.shape
|
|
||||||
yuv_flat = yuv_array.reshape(-1, 3)
|
|
||||||
|
|
||||||
# Apply transformation
|
|
||||||
rgb_flat = yuv_flat @ yuv_to_rgb_matrix.T
|
|
||||||
|
|
||||||
# Reshape back
|
|
||||||
rgb_array = rgb_flat.reshape(original_shape)
|
|
||||||
|
|
||||||
return rgb_array
|
|
||||||
|
|
||||||
def erase_rainbow_artifacts(img, is_color):
|
|
||||||
"""
|
|
||||||
Remove rainbow artifacts from grayscale or color images.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
img: PIL Image (grayscale or RGB)
|
|
||||||
is_color: Boolean indicating if the image is to be treated as color
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
PIL.Image: Cleaned image
|
|
||||||
"""
|
|
||||||
# Auto-detect color mode if not specified
|
|
||||||
if is_color is None:
|
|
||||||
color = img.mode in ('RGB', 'RGBA', 'L') and len(np.array(img).shape) == 3
|
|
||||||
|
|
||||||
if is_color and img.mode in ('RGB', 'RGBA'):
|
|
||||||
# Convert to RGB if needed
|
|
||||||
if img.mode == 'RGBA':
|
|
||||||
img = img.convert('RGB')
|
|
||||||
|
|
||||||
# Convert to numpy array
|
|
||||||
img_array = np.array(img, dtype=np.float32)
|
|
||||||
|
|
||||||
# Convert to YUV color space
|
|
||||||
yuv_array = rgb_to_yuv(img_array)
|
|
||||||
|
|
||||||
# Extract luminance channel (Y)
|
|
||||||
luminance = yuv_array[:, :, 0]
|
|
||||||
|
|
||||||
# Process only the luminance channel
|
|
||||||
fft_spectrum = fourier_transform_image(luminance)
|
|
||||||
clean_spectrum = attenuate_diagonal_frequencies(fft_spectrum)
|
|
||||||
clean_luminance = np.fft.irfft2(clean_spectrum, s=luminance.shape)
|
|
||||||
|
|
||||||
# Normalize and clip luminance
|
|
||||||
clean_luminance = np.clip(clean_luminance, 0, 255)
|
|
||||||
|
|
||||||
# Replace luminance in YUV array
|
|
||||||
yuv_array[:, :, 0] = clean_luminance
|
|
||||||
|
|
||||||
# Convert back to RGB
|
|
||||||
rgb_array = yuv_to_rgb(yuv_array)
|
|
||||||
rgb_array = np.clip(rgb_array, 0, 255).astype(np.uint8)
|
|
||||||
|
|
||||||
# Convert back to PIL image
|
|
||||||
clean_image = Image.fromarray(rgb_array, mode='RGB')
|
|
||||||
|
|
||||||
else:
|
|
||||||
# Grayscale processing (original behavior)
|
|
||||||
if img.mode != 'L':
|
|
||||||
img = img.convert('L')
|
|
||||||
|
|
||||||
# Get original image dimensions
|
|
||||||
original_shape = (img.height, img.width)
|
|
||||||
|
|
||||||
fft_spectrum = fourier_transform_image(img)
|
|
||||||
clean_spectrum = attenuate_diagonal_frequencies(fft_spectrum)
|
|
||||||
clean_image = inverse_fourier_transform_image(clean_spectrum, is_color, original_shape)
|
|
||||||
|
|
||||||
return clean_image
|
|
||||||
@@ -19,6 +19,7 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
import os
|
import os
|
||||||
|
from hashlib import md5
|
||||||
from html.parser import HTMLParser
|
from html.parser import HTMLParser
|
||||||
import subprocess
|
import subprocess
|
||||||
from packaging.version import Version
|
from packaging.version import Version
|
||||||
@@ -27,9 +28,6 @@ import sys
|
|||||||
from traceback import format_tb
|
from traceback import format_tb
|
||||||
|
|
||||||
|
|
||||||
IMAGE_TYPES = ('.png', '.jpg', '.jpeg', '.gif', '.webp', '.jp2', '.avif')
|
|
||||||
|
|
||||||
|
|
||||||
class HTMLStripper(HTMLParser):
|
class HTMLStripper(HTMLParser):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
HTMLParser.__init__(self)
|
HTMLParser.__init__(self)
|
||||||
@@ -48,17 +46,11 @@ class HTMLStripper(HTMLParser):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
def dot_clean(filetree):
|
|
||||||
for root, _, files in os.walk(filetree, topdown=False):
|
|
||||||
for name in files:
|
|
||||||
if name.startswith('._') or name == '.DS_Store':
|
|
||||||
if os.path.exists(os.path.join(root, name)):
|
|
||||||
os.remove(os.path.join(root, name))
|
|
||||||
|
|
||||||
|
|
||||||
def getImageFileName(imgfile):
|
def getImageFileName(imgfile):
|
||||||
name, ext = os.path.splitext(imgfile)
|
name, ext = os.path.splitext(imgfile)
|
||||||
ext = ext.lower()
|
ext = ext.lower()
|
||||||
|
if (name.startswith('.') and len(name) == 1) or ext not in ['.png', '.jpg', '.jpeg', '.gif', '.webp']:
|
||||||
|
return None
|
||||||
return [name, ext]
|
return [name, ext]
|
||||||
|
|
||||||
|
|
||||||
@@ -82,6 +74,16 @@ def walkLevel(some_dir, level=1):
|
|||||||
del dirs[:]
|
del dirs[:]
|
||||||
|
|
||||||
|
|
||||||
|
def md5Checksum(fpath):
|
||||||
|
with open(fpath, 'rb') as fh:
|
||||||
|
m = md5()
|
||||||
|
while True:
|
||||||
|
data = fh.read(8192)
|
||||||
|
if not data:
|
||||||
|
break
|
||||||
|
m.update(data)
|
||||||
|
return m.hexdigest()
|
||||||
|
|
||||||
|
|
||||||
def sanitizeTrace(traceback):
|
def sanitizeTrace(traceback):
|
||||||
return ''.join(format_tb(traceback))\
|
return ''.join(format_tb(traceback))\
|
||||||
@@ -101,10 +103,10 @@ def dependencyCheck(level):
|
|||||||
if level > 2:
|
if level > 2:
|
||||||
try:
|
try:
|
||||||
from PySide6.QtCore import qVersion as qtVersion
|
from PySide6.QtCore import qVersion as qtVersion
|
||||||
if Version('6.0.0') > Version(qtVersion()):
|
if Version('6.5.1') > Version(qtVersion()):
|
||||||
missing.append('PySide 6.0.0')
|
missing.append('PySide 6.5.1+')
|
||||||
except ImportError:
|
except ImportError:
|
||||||
missing.append('PySide 6.0.0+')
|
missing.append('PySide 6.5.1+')
|
||||||
try:
|
try:
|
||||||
import raven
|
import raven
|
||||||
except ImportError:
|
except ImportError:
|
||||||
@@ -127,16 +129,10 @@ def dependencyCheck(level):
|
|||||||
missing.append('python-slugify 1.2.1+')
|
missing.append('python-slugify 1.2.1+')
|
||||||
try:
|
try:
|
||||||
from PIL import __version__ as pillowVersion
|
from PIL import __version__ as pillowVersion
|
||||||
if Version('8.3.0') > Version(pillowVersion):
|
if Version('5.2.0') > Version(pillowVersion):
|
||||||
missing.append('Pillow 8.3.0+')
|
missing.append('Pillow 5.2.0+')
|
||||||
except ImportError:
|
except ImportError:
|
||||||
missing.append('Pillow 8.3.0+')
|
missing.append('Pillow 5.2.0+')
|
||||||
try:
|
|
||||||
from pymupdf import __version__ as pymupdfVersion
|
|
||||||
if Version('1.16.1') > Version(pymupdfVersion):
|
|
||||||
missing.append('PyMuPDF 1.16.1+')
|
|
||||||
except ImportError:
|
|
||||||
missing.append('PyMuPDF 1.16.1+')
|
|
||||||
if len(missing) > 0:
|
if len(missing) > 0:
|
||||||
print('ERROR: ' + ', '.join(missing) + ' is not installed!')
|
print('ERROR: ' + ', '.join(missing) + ' is not installed!')
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|||||||
@@ -1,11 +0,0 @@
|
|||||||
Pillow>=11.3.0
|
|
||||||
psutil>=5.9.5
|
|
||||||
requests>=2.31.0
|
|
||||||
python-slugify>=1.2.1
|
|
||||||
packaging>=23.2
|
|
||||||
mozjpeg-lossless-optimization>=1.2.0
|
|
||||||
natsort>=8.4.0
|
|
||||||
distro>=1.8.0
|
|
||||||
# Below requirements are compiled in Dockefile
|
|
||||||
# numpy==2.3.4
|
|
||||||
# PyMuPDF==1.26.6
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
PySide6==6.4.3
|
|
||||||
Pillow>=11.3.0
|
|
||||||
psutil>=5.9.5
|
|
||||||
requests>=2.31.0
|
|
||||||
python-slugify>=1.2.1
|
|
||||||
raven>=6.0.0
|
|
||||||
packaging>=23.2
|
|
||||||
mozjpeg-lossless-optimization>=1.2.0
|
|
||||||
natsort>=8.4.0
|
|
||||||
distro>=1.8.0
|
|
||||||
numpy<2
|
|
||||||
PyMuPDF>=1.26.1
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
PySide6==6.1.3
|
|
||||||
Pillow>=9
|
|
||||||
psutil>=5.9.5
|
|
||||||
requests>=2.31.0
|
|
||||||
python-slugify>=1.2.1
|
|
||||||
raven>=6.0.0
|
|
||||||
packaging>=23.2
|
|
||||||
mozjpeg-lossless-optimization>=1.2.0
|
|
||||||
natsort>=8.4.0
|
|
||||||
distro>=1.8.0
|
|
||||||
numpy==1.23.0
|
|
||||||
PyMuPDF>=1.16
|
|
||||||
@@ -1,12 +1,11 @@
|
|||||||
PySide6>6
|
PySide6>=6.5.1
|
||||||
Pillow>=11.3.0
|
Pillow>=5.2.0
|
||||||
psutil>=5.9.5
|
psutil>=5.9.5
|
||||||
requests>=2.31.0
|
requests>=2.31.0
|
||||||
python-slugify>=1.2.1,<9.0.0
|
python-slugify>=1.2.1
|
||||||
raven>=6.0.0
|
raven>=6.0.0
|
||||||
packaging>=23.2
|
packaging>=23.2
|
||||||
mozjpeg-lossless-optimization>=1.2.0
|
mozjpeg-lossless-optimization>=1.1.2
|
||||||
natsort>=8.4.0
|
natsort>=8.4.0
|
||||||
distro>=1.8.0
|
distro>=1.8.0
|
||||||
numpy>=1.22.4
|
numpy>=1.22.4,<2.0.0
|
||||||
PyMuPDF>=1.18.0
|
|
||||||
|
|||||||
91
setup.py
91
setup.py
@@ -8,8 +8,6 @@ Install as Python package:
|
|||||||
|
|
||||||
Create EXE/APP:
|
Create EXE/APP:
|
||||||
python3 setup.py build_binary
|
python3 setup.py build_binary
|
||||||
python3 setup.py build_c2e
|
|
||||||
python3 setup.py build_c2p
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import os
|
import os
|
||||||
@@ -40,17 +38,10 @@ class BuildBinaryCommand(setuptools.Command):
|
|||||||
if sys.platform == 'darwin':
|
if sys.platform == 'darwin':
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -D -i icons/comic2ebook.icns -n "Kindle Comic Converter" -w -s kcc.py')
|
os.system('pyinstaller --hidden-import=_cffi_backend -y -D -i icons/comic2ebook.icns -n "Kindle Comic Converter" -w -s kcc.py')
|
||||||
# TODO /usr/bin/codesign --force -s "$MACOS_CERTIFICATE_NAME" --options runtime dist/Applications/Kindle\ Comic\ Converter.app -v
|
# TODO /usr/bin/codesign --force -s "$MACOS_CERTIFICATE_NAME" --options runtime dist/Applications/Kindle\ Comic\ Converter.app -v
|
||||||
min_os = os.getenv('MACOSX_DEPLOYMENT_TARGET', '')
|
os.system(f'appdmg kcc.json dist/kcc_macos_{platform.processor()}_{VERSION}.dmg')
|
||||||
if min_os.startswith('10.1'):
|
|
||||||
os.system(f'appdmg kcc.json dist/kcc_osx_{min_os.replace(".", "_")}_legacy_{VERSION}.dmg')
|
|
||||||
else:
|
|
||||||
os.system(f'appdmg kcc.json dist/kcc_macos_{platform.processor()}_{VERSION}.dmg')
|
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
elif sys.platform == 'win32':
|
elif sys.platform == 'win32':
|
||||||
if os.getenv('WINDOWS_7'):
|
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n KCC_' + VERSION + ' -w --noupx kcc.py')
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n kcc_win7_legacy_' + VERSION + ' -w --noupx kcc.py')
|
|
||||||
else:
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n KCC_' + VERSION + ' -w --noupx kcc.py')
|
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
elif sys.platform == 'linux':
|
elif sys.platform == 'linux':
|
||||||
os.system(
|
os.system(
|
||||||
@@ -59,75 +50,10 @@ class BuildBinaryCommand(setuptools.Command):
|
|||||||
else:
|
else:
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
# noinspection PyUnresolvedReferences
|
|
||||||
class BuildC2ECommand(setuptools.Command):
|
|
||||||
description = 'build binary c2e release'
|
|
||||||
user_options = []
|
|
||||||
|
|
||||||
def initialize_options(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def finalize_options(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
# noinspection PyShadowingNames
|
|
||||||
def run(self):
|
|
||||||
VERSION = __version__
|
|
||||||
if sys.platform == 'darwin':
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -D -i icons/comic2ebook.icns -n "KCC C2E" -c -s kcc-c2e.py')
|
|
||||||
# TODO /usr/bin/codesign --force -s "$MACOS_CERTIFICATE_NAME" --options runtime dist/Applications/Kindle\ Comic\ Converter.app -v
|
|
||||||
sys.exit(0)
|
|
||||||
elif sys.platform == 'win32':
|
|
||||||
if os.getenv('WINDOWS_7'):
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n kcc_c2e_win7_legacy_' + VERSION + ' -c --noupx kcc-c2e.py')
|
|
||||||
else:
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n kcc_c2e_' + VERSION + ' -c --noupx kcc-c2e.py')
|
|
||||||
sys.exit(0)
|
|
||||||
elif sys.platform == 'linux':
|
|
||||||
os.system(
|
|
||||||
'pyinstaller --hidden-import=_cffi_backend --hidden-import=queue -y -F -i icons/comic2ebook.ico -n kcc_c2e_linux_' + VERSION + ' kcc-c2e.py')
|
|
||||||
sys.exit(0)
|
|
||||||
else:
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnresolvedReferences
|
|
||||||
class BuildC2PCommand(setuptools.Command):
|
|
||||||
description = 'build binary c2p release'
|
|
||||||
user_options = []
|
|
||||||
|
|
||||||
def initialize_options(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def finalize_options(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
# noinspection PyShadowingNames
|
|
||||||
def run(self):
|
|
||||||
VERSION = __version__
|
|
||||||
if sys.platform == 'darwin':
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -n "KCC C2P" -c -s kcc-c2p.py')
|
|
||||||
# TODO /usr/bin/codesign --force -s "$MACOS_CERTIFICATE_NAME" --options runtime dist/Applications/Kindle\ Comic\ Converter.app -v
|
|
||||||
sys.exit(0)
|
|
||||||
elif sys.platform == 'win32':
|
|
||||||
if os.getenv('WINDOWS_7'):
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n kcc_c2p_win7_legacy_' + VERSION + ' -c --noupx kcc-c2p.py')
|
|
||||||
else:
|
|
||||||
os.system('pyinstaller --hidden-import=_cffi_backend -y -F -i icons\\comic2ebook.ico -n kcc_c2p_' + VERSION + ' -c --noupx kcc-c2p.py')
|
|
||||||
sys.exit(0)
|
|
||||||
elif sys.platform == 'linux':
|
|
||||||
os.system(
|
|
||||||
'pyinstaller --hidden-import=_cffi_backend --hidden-import=queue -y -F -i icons/comic2ebook.ico -n kcc_c2p_linux_' + VERSION + ' kcc-c2p.py')
|
|
||||||
sys.exit(0)
|
|
||||||
else:
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
|
|
||||||
setuptools.setup(
|
setuptools.setup(
|
||||||
cmdclass={
|
cmdclass={
|
||||||
'build_binary': BuildBinaryCommand,
|
'build_binary': BuildBinaryCommand,
|
||||||
'build_c2e': BuildC2ECommand,
|
|
||||||
'build_c2p': BuildC2PCommand,
|
|
||||||
},
|
},
|
||||||
name=NAME,
|
name=NAME,
|
||||||
version=VERSION,
|
version=VERSION,
|
||||||
@@ -148,17 +74,16 @@ setuptools.setup(
|
|||||||
},
|
},
|
||||||
packages=['kindlecomicconverter'],
|
packages=['kindlecomicconverter'],
|
||||||
install_requires=[
|
install_requires=[
|
||||||
'PySide6>=6.0.0',
|
'pyside6>=6.5.1',
|
||||||
'Pillow>=9.3.0',
|
'Pillow>=5.2.0',
|
||||||
'psutil>=5.9.5',
|
'psutil>=5.9.5',
|
||||||
'requests>=2.31.0',
|
|
||||||
'python-slugify>=1.2.1,<9.0.0',
|
'python-slugify>=1.2.1,<9.0.0',
|
||||||
'raven>=6.0.0',
|
'raven>=6.0.0',
|
||||||
'mozjpeg-lossless-optimization>=1.2.0',
|
'requests>=2.31.0',
|
||||||
|
'mozjpeg-lossless-optimization>=1.1.2',
|
||||||
'natsort>=8.4.0',
|
'natsort>=8.4.0',
|
||||||
'distro>=1.8.0',
|
'distro',
|
||||||
'numpy>=1.22.4',
|
'numpy>=1.22.4,<2.0.0'
|
||||||
'PyMuPDF>=1.16.1',
|
|
||||||
],
|
],
|
||||||
classifiers=[],
|
classifiers=[],
|
||||||
zip_safe=False,
|
zip_safe=False,
|
||||||
|
|||||||
Reference in New Issue
Block a user