diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md index 5a28406a..a4621025 100644 --- a/.github/ISSUE_TEMPLATE/bug_report.md +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -11,7 +11,7 @@ assignees: '' After 6 years of more or less intensive programming on Calibre-Web, I need a break. The last few months, maintaining Calibre-Web has felt more like work than a hobby. I felt pressured and teased by people to solve "their" problems and merge PRs for "their" Calibre-Web. -I have turned off all notifications from Github/Discord and will now concentrate undisturbed on the development of “my” Calibre-Web over the next few weeks/months. +I have turned off all notifications from GitHub/Discord and will now concentrate undisturbed on the development of “my” Calibre-Web over the next few weeks/months. I will look into the issues and maybe also the PRs from time to time, but don't expect a quick response from me. diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md index dec77d51..3e5c7b9f 100644 --- a/.github/ISSUE_TEMPLATE/feature_request.md +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -11,7 +11,7 @@ assignees: '' After 6 years of more or less intensive programming on Calibre-Web, I need a break. The last few months, maintaining Calibre-Web has felt more like work than a hobby. I felt pressured and teased by people to solve "their" problems and merge PRs for "their" Calibre-Web. -I have turned off all notifications from Github/Discord and will now concentrate undisturbed on the development of “my” Calibre-Web over the next few weeks/months. +I have turned off all notifications from GitHub/Discord and will now concentrate undisturbed on the development of “my” Calibre-Web over the next few weeks/months. I will look into the issues and maybe also the PRs from time to time, but don't expect a quick response from me. Please have a look at our [Contributing Guidelines](https://github.com/janeczku/calibre-web/blob/master/CONTRIBUTING.md) diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 2ab32cf9..5f538dfa 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -20,7 +20,7 @@ Some of the user languages in Calibre-Web having missing translations. We are ha ### **Documentation** -The Calibre-Web documentation is hosted in the Github [Wiki](https://github.com/janeczku/calibre-web/wiki). The Wiki is open to everybody, if you find a problem, feel free to correct it. If information is missing, you are welcome to add it. The content will be reviewed time by time. Please try to be consistent with the form with the other Wiki pages (e.g. the project name is Calibre-Web with 2 capital letters and a dash in between). +The Calibre-Web documentation is hosted in the GitHub [Wiki](https://github.com/janeczku/calibre-web/wiki). The Wiki is open to everybody, if you find a problem, feel free to correct it. If information is missing, you are welcome to add it. The content will be reviewed time by time. Please try to be consistent with the form with the other Wiki pages (e.g. the project name is Calibre-Web with 2 capital letters and a dash in between). ### **Reporting a bug** @@ -28,12 +28,12 @@ Do not open up a GitHub issue if the bug is a **security vulnerability** in Cali Ensure the **bug was not already reported** by searching on GitHub under [Issues](https://github.com/janeczku/calibre-web/issues). Please also check if a solution for your problem can be found in the [wiki](https://github.com/janeczku/calibre-web/wiki). -If you're unable to find an **open issue** addressing the problem, open a [new one](https://github.com/janeczku/calibre-web/issues/new/choose). Be sure to include a **title** and **clear description**, as much relevant information as possible, the **issue form** helps you providing the right information. Deleting the form and just pasting the stack trace doesn't speed up fixing the problem. If your issue could be resolved, consider closing the issue. +If you're unable to find an **open issue** addressing the problem, open a [new one](https://github.com/janeczku/calibre-web/issues/new/choose). Be sure to include a **title** and **clear description**, as much relevant information as possible, the **issue form** helps you provide the right information. Deleting the form and just pasting the stack trace doesn't speed up fixing the problem. If your issue could be resolved, consider closing the issue. ### **Feature Request** If there is a feature missing in Calibre-Web and you can't find a feature request in the [Issues](https://github.com/janeczku/calibre-web/issues) section, you could create a [feature request](https://github.com/janeczku/calibre-web/issues/new?assignees=&labels=&template=feature_request.md&title=). -We will not extend Calibre-Web with any more login abilities or add further files storages, or file syncing ability. Furthermore Calibre-Web is made for home usage for company in-house usage, so requests regarding any sorts of social interaction capability, payment routines, search engine or web site analytics integration will not be implemented. +We will not extend Calibre-Web with any more login abilities or add further files storages, or file syncing ability. Calibre-Web is made for home usage for company in-house usage, so requests regarding any sorts of social interaction capability, payment routines, search engine or website analytics integration will not be implemented. ### **Contributing code to Calibre-Web** @@ -42,5 +42,5 @@ Open a new GitHub pull request with the patch. Ensure the PR description clearly In case your code enhances features of Calibre-Web: Create your pull request for the development branch if your enhancement consists of more than some lines of code in a local section of Calibre-Webs code. This makes it easier to test it and check all implication before it's made public. Please check if your code runs with python 3, python 2 is no longer supported. If possible and the feature is related to operating system functions, try to check it on Windows and Linux. -Calibre-Web is automatically tested on Linux in combination with python 3.8. The code for testing is in a [separate repo](https://github.com/OzzieIsaacs/calibre-web-test) on Github. It uses unit tests and performs real system tests with selenium; it would be great if you could consider also writing some tests. +Calibre-Web is automatically tested on Linux in combination with python 3.8. The code for testing is in a [separate repo](https://github.com/OzzieIsaacs/calibre-web-test) on GitHub. It uses unit tests and performs real system tests with selenium; it would be great if you could consider also writing some tests. A static code analysis is done by Codacy, but it's partly broken and doesn't run automatically. You could check your code with ESLint before contributing, a configuration file can be found in the projects root folder. diff --git a/MANIFEST.in b/MANIFEST.in index b667159c..f07c4d83 100644 --- a/MANIFEST.in +++ b/MANIFEST.in @@ -1 +1,3 @@ graft src/calibreweb +global-exclude __pycache__ +global-exclude *.pyc diff --git a/README.md b/README.md index a400b3b2..2d9c7d18 100755 --- a/README.md +++ b/README.md @@ -1,10 +1,3 @@ -# Short Notice from the maintainer - -After 6 years of more or less intensive programming on Calibre-Web, I need a break. -The last few months, maintaining Calibre-Web has felt more like work than a hobby. I felt pressured and teased by people to solve "their" problems and merge PRs for "their" Calibre-Web. -I have turned off all notifications from Github/Discord and will now concentrate undisturbed on the development of “my” Calibre-Web over the next few weeks/months. -I will look into the issues and maybe also the PRs from time to time, but don't expect a quick response from me. - # Calibre-Web Calibre-Web is a web app that offers a clean and intuitive interface for browsing, reading, and downloading eBooks using a valid [Calibre](https://calibre-ebook.com) database. @@ -26,13 +19,13 @@ Calibre-Web is a web app that offers a clean and intuitive interface for browsin - [Quick start](#quick-start) - [Requirements](#requirements) 4. [Docker Images](#docker-images) -5. [Contributor Recognition](#contributor-recognition) -6. [Contact](#contact) -7. [Contributing to Calibre-Web](#contributing-to-calibre-web) +5. [Troubleshooting](#troubleshooting) +6. [Contributor Recognition](#contributor-recognition) +7. [Contact](#contact) +8. [Contributing to Calibre-Web](#contributing-to-calibre-web) - *This software is a fork of [library](https://github.com/mutschler/calibreserver) and licensed under the GPL v3 License.* ![Main screen](https://github.com/janeczku/calibre-web/wiki/images/main_screen.png) @@ -64,52 +57,102 @@ Calibre-Web is a web app that offers a clean and intuitive interface for browsin ## Installation -#### Installation via pip (recommended) -1. Create a virtual environment for Calibre-Web to avoid conflicts with existing Python dependencies -2. Install Calibre-Web via pip: `pip install calibreweb` (or `pip3` depending on your OS/distro) -3. Install optional features via pip as needed, see [this page](https://github.com/janeczku/calibre-web/wiki/Dependencies-in-Calibre-Web-Linux-and-Windows) for details -4. Start Calibre-Web by typing `cps` +### Installation via pip (recommended) -*Note: Raspberry Pi OS users may encounter issues during installation. If so, please update pip (`./venv/bin/python3 -m pip install --upgrade pip`) and/or install cargo (`sudo apt install cargo`) before retrying the installation.* +1. **Create a virtual environment**: It’s essential to isolate your Calibre-Web installation to avoid dependency conflicts. You can create a virtual environment by running: + ``` + python3 -m venv calibre-web-env + ``` +2. **Activate the virtual environment**: + ``` + source calibre-web-env/bin/activate + ``` +3. **Install Calibre-Web**: Use pip to install the application: + ``` + pip install calibreweb + ``` +4. **Install optional features**: For additional functionality, you may need to install optional features. Refer to [this page](https://github.com/janeczku/calibre-web/wiki/Dependencies-in-Calibre-Web-Linux-and-Windows) for details on what can be installed. +5. **Start Calibre-Web**: After installation, you can start the application with: + ``` + cps + ``` -Refer to the Wiki for additional installation examples: [manual installation](https://github.com/janeczku/calibre-web/wiki/Manual-installation), [Linux Mint](https://github.com/janeczku/calibre-web/wiki/How-To:-Install-Calibre-Web-in-Linux-Mint-19-or-20), [Cloud Provider](https://github.com/janeczku/calibre-web/wiki/How-To:-Install-Calibre-Web-on-a-Cloud-Provider). +*Note: Users of Raspberry Pi OS may encounter installation issues. If you do, try upgrading pip and/or installing cargo as follows:* + ``` + ./venv/bin/python3 -m pip install --upgrade pip + sudo apt install cargo + ``` + +### Important Links +- For additional installation examples, check the following: + - [Manual installation](https://github.com/janeczku/calibre-web/wiki/Manual-installation) + - [Linux Mint installation](https://github.com/janeczku/calibre-web/wiki/How-To:-Install-Calibre-Web-in-Linux-Mint-19-or-20) + - [Cloud Provider setup](https://github.com/janeczku/calibre-web/wiki/How-To:-Install-Calibre-Web-on-a-Cloud-Provider) ## Quick Start -1. Open your browser and navigate to `http://localhost:8083` or `http://localhost:8083/opds` for the OPDS catalog -2. Log in with the default admin credentials -3. If you don't have a Calibre database, you can use [this database](https://github.com/janeczku/calibre-web/raw/master/library/metadata.db) (move it out of the Calibre-Web folder to prevent overwriting during updates) -4. Set `Location of Calibre database` to the path of the folder containing your Calibre library (metadata.db) and click "Save" -5. Optionally, use Google Drive to host your Calibre library by following the [Google Drive integration guide](https://github.com/janeczku/calibre-web/wiki/G-Drive-Setup#using-google-drive-integration) -6. Configure your Calibre-Web instance via the admin page, referring to the [Basic Configuration](https://github.com/janeczku/calibre-web/wiki/Configuration#basic-configuration) and [UI Configuration](https://github.com/janeczku/calibre-web/wiki/Configuration#ui-configuration) guides - -#### Default Admin Login: -- **Username:** admin -- **Password:** admin123 +1. **Access Calibre-Web**: Open your browser and navigate to: + ``` + http://localhost:8083 + ``` + or for the OPDS catalog: + ``` + http://localhost:8083/opds + ``` +2. **Log in**: Use the default admin credentials: + - **Username:** admin + - **Password:** admin123 +3. **Database Setup**: If you do not have a Calibre database, download a sample from: + ``` + https://github.com/janeczku/calibre-web/raw/master/library/metadata.db + ``` + Move it out of the Calibre-Web folder to avoid overwriting during updates. +4. **Configure Calibre Database**: In the admin interface, set the `Location of Calibre database` to the path of the folder containing your Calibre library (where `metadata.db` is located) and click "Save". +5. **Google Drive Integration**: For hosting your Calibre library on Google Drive, refer to the [Google Drive integration guide](https://github.com/janeczku/calibre-web/wiki/G-Drive-Setup#using-google-drive-integration). +6. **Admin Configuration**: Configure your instance via the admin page, referring to the [Basic Configuration](https://github.com/janeczku/calibre-web/wiki/Configuration#basic-configuration) and [UI Configuration](https://github.com/janeczku/calibre-web/wiki/Configuration#ui-configuration) guides. ## Requirements -- Python 3.7+ -- [Imagemagick](https://imagemagick.org/script/download.php) for cover extraction from EPUBs (Windows users may need to install [Ghostscript](https://ghostscript.com/releases/gsdnld.html) for PDF cover extraction) -- Windows users need to install [libmagic for 32bit python](https://gnuwin32.sourceforge.net/downlinks/file.php) or [libmagic for 64bit python](https://github.com/nscaife/file-windows/releases/tag/20170108), depending on the python version; The files need to be installed in path (e.g. script folder of your Calibre-Web venv, or in the root folder of Calibre-Web -- Optional: [Calibre desktop program](https://calibre-ebook.com/download) for on-the-fly conversion and metadata editing (set "calibre's converter tool" path on the setup page) -- Optional: [Kepubify tool](https://github.com/pgaskin/kepubify/releases/latest) for Kobo device support (place the binary in `/opt/kepubify` on Linux or `C:\Program Files\kepubify` on Windows) +- **Python Version**: Ensure you have Python 3.7 or newer. +- **Imagemagick**: Required for cover extraction from EPUBs. Windows users may also need to install [Ghostscript](https://ghostscript.com/releases/gsdnld.html) for PDF cover extraction. +- **Optional Tools**: + - **Calibre desktop program**: Recommended for on-the-fly conversion and metadata editing. Set the path to Calibre’s converter tool on the setup page. + - **Kepubify tool**: Needed for Kobo device support. Download the tool and place the binary in `/opt/kepubify` on Linux or `C:\Program Files\kepubify` on Windows. ## Docker Images -Pre-built Docker images are available in the following Docker Hub repositories (maintained by the LinuxServer team): +Pre-built Docker images are available: -#### **LinuxServer - x64, aarch64** -- [Docker Hub](https://hub.docker.com/r/linuxserver/calibre-web) -- [GitHub](https://github.com/linuxserver/docker-calibre-web) -- [GitHub - Optional Calibre layer](https://github.com/linuxserver/docker-mods/tree/universal-calibre) +### **LinuxServer - x64, aarch64** +- **Docker Hub**: [linuxserver/calibre-web](https://hub.docker.com/r/linuxserver/calibre-web) +- **GitHub**: [linuxserver/docker-calibre-web](https://github.com/linuxserver/docker-calibre-web) +- **Optional Calibre layer**: [linuxserver/docker-mods](https://github.com/linuxserver/docker-mods/tree/universal-calibre) - Include the environment variable `DOCKER_MODS=linuxserver/mods:universal-calibre` in your Docker run/compose file to add the Calibre `ebook-convert` binary (x64 only). Omit this variable for a lightweight image. +To include the Calibre `ebook-convert` binary (x64 only), add the environment variable: +``` +DOCKER_MODS=linuxserver/mods:universal-calibre +``` +in your Docker run/compose file. Omit this variable for a lightweight image. - Both the Calibre-Web and Calibre-Mod images are automatically rebuilt on new releases and updates. +- **Paths Configuration**: + - Set **Path to Calibre Binaries** to `/usr/bin`. + - Set **Path to Unrar** to `/usr/bin/unrar`. - - Set "Path to Calibre Binaries" to `/usr/bin` - - Set "Path to Unrar" to `/usr/bin/unrar` +## Troubleshooting + +- **Common Issues**: + - If you experience issues starting the application, check the log files located in the `logs` directory for error messages. + - If eBooks fail to load, verify that the `Location of Calibre database` is correctly set and that the database file is accessible. + +- **Configuration Errors**: Ensure that your Calibre database is compatible and properly formatted. Refer to the Calibre documentation for guidance on maintaining the database. + +- **Performance Problems**: + - If the application is slow, consider increasing the allocated resources (CPU/RAM) to your server or optimizing the Calibre database by removing duplicates and unnecessary entries. + - Regularly clear the cache in your web browser to improve loading times. + +- **User Management Issues**: If users are unable to log in or register, check the user permission settings in the admin interface. Ensure that registration is enabled and that users are being assigned appropriate roles. + +- **Support Resources**: For additional help, consider visiting the [FAQ section](https://github.com/janeczku/calibre-web/wiki/FAQ) of the wiki or posting your questions in the [Discord community](https://discord.gg/h2VsJ2NEfB). ## Contributor Recognition @@ -123,4 +166,21 @@ For more information, How To's, and FAQs, please visit the [Wiki](https://github ## Contributing to Calibre-Web -Check out our [Contributing Guidelines](https://github.com/janeczku/calibre-web/blob/master/CONTRIBUTING.md) +To contribute, please check our [Contributing Guidelines](https://github.com/janeczku/calibre-web/blob/master/CONTRIBUTING.md). We welcome issues, feature requests, and pull requests from the community. + +### Reporting Bugs + +If you encounter bugs or issues, please report them in the [issues section](https://github.com/janeczku/calibre-web/issues) of the repository. Be sure to include detailed information about your setup and the problem encountered. + +### Feature Requests + +We welcome suggestions for new features. Please create a new issue in the repository to discuss your ideas. + +## Additional Resources + +- **Documentation**: Comprehensive documentation is available on the [Calibre-Web wiki](https://github.com/janeczku/calibre-web/wiki). +- **Community Contributions**: Explore the [community contributions](https://github.com/janeczku/calibre-web/pulls) to see ongoing work and how you can get involved. + +--- + +Thank you for using Calibre-Web! We hope you enjoy managing your eBook library with our tool. diff --git a/SECURITY.md b/SECURITY.md index e4ab1a8d..c0861a18 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -10,41 +10,46 @@ To receive fixes for security vulnerabilities it is required to always upgrade t ## History -| Fixed in | Description |CVE number | -|---------------|--------------------------------------------------------------------------------------------------------------------|---------| -| 3rd July 2018 | Guest access acts as a backdoor || -| V 0.6.7 | Hardcoded secret key for sessions |CVE-2020-12627 | -| V 0.6.13 | Calibre-Web Metadata cross site scripting |CVE-2021-25964| -| V 0.6.13 | Name of Shelves are only visible to users who can access the corresponding shelf Thanks to @ibarrionuevo || -| V 0.6.13 | JavaScript could get executed in the description field. Thanks to @ranjit-git and Hagai Wechsler (WhiteSource) || -| V 0.6.13 | JavaScript could get executed in a custom column of type "comment" field || -| V 0.6.13 | JavaScript could get executed after converting a book to another format with a title containing javascript code || -| V 0.6.13 | JavaScript could get executed after converting a book to another format with a username containing javascript code || -| V 0.6.13 | JavaScript could get executed in the description series, categories or publishers title || -| V 0.6.13 | JavaScript could get executed in the shelf title || -| V 0.6.13 | Login with the old session cookie after logout. Thanks to @ibarrionuevo || -| V 0.6.14 | CSRF was possible. Thanks to @mik317 and Hagai Wechsler (WhiteSource) |CVE-2021-25965| -| V 0.6.14 | Migrated some routes to POST-requests (CSRF protection). Thanks to @scara31 |CVE-2021-4164| -| V 0.6.15 | Fix for "javascript:" script links in identifier. Thanks to @scara31 |CVE-2021-4170| -| V 0.6.15 | Cross-Site Scripting vulnerability on uploaded cover file names. Thanks to @ibarrionuevo || -| V 0.6.15 | Creating public shelfs is now denied if user is missing the edit public shelf right. Thanks to @ibarrionuevo || -| V 0.6.15 | Changed error message in case of trying to delete a shelf unauthorized. Thanks to @ibarrionuevo || -| V 0.6.16 | JavaScript could get executed on authors page. Thanks to @alicaz |CVE-2022-0352| -| V 0.6.16 | Localhost can no longer be used to upload covers. Thanks to @scara31 |CVE-2022-0339| -| V 0.6.16 | Another case where public shelfs could be created without permission is prevented. Thanks to @nhiephon |CVE-2022-0273| -| V 0.6.16 | It's prevented to get the name of a private shelfs. Thanks to @nhiephon |CVE-2022-0405| -| V 0.6.17 | The SSRF Protection can no longer be bypassed via an HTTP redirect. Thanks to @416e6e61 |CVE-2022-0767| -| V 0.6.17 | The SSRF Protection can no longer be bypassed via 0.0.0.0 and it's ipv6 equivalent. Thanks to @r0hanSH |CVE-2022-0766| -| V 0.6.18 | Possible SQL Injection is prevented in user table Thanks to Iman Sharafaldin (Forward Security) |CVE-2022-30765| -| V 0.6.18 | The SSRF protection no longer can be bypassed by IPV6/IPV4 embedding. Thanks to @416e6e61 |CVE-2022-0939| -| V 0.6.18 | The SSRF protection no longer can be bypassed to connect to other servers in the local network. Thanks to @michaellrowley |CVE-2022-0990| -| V 0.6.20 | Credentials for emails are now stored encrypted || -| V 0.6.20 | Login is rate limited || -| V 0.6.20 | Passwordstrength can be forced || -| V 0.6.21 | SMTP server credentials are no longer returned to client || -| V 0.6.21 | Cross-site scripting (XSS) stored in href bypasses filter using data wrapper no longer possible || -| V 0.6.21 | Cross-site scripting (XSS) is no longer possible via pathchooser || -| V 0.6.21 | Error Handling at non existent rating, language, and user downloaded books was fixed || +| Fixed in | Description | CVE number | +|---------------|----------------------------------------------------------------------------------------------------------------------------------|----------------| +| 3rd July 2018 | Guest access acts as a backdoor | | +| V 0.6.7 | Hardcoded secret key for sessions | CVE-2020-12627 | +| V 0.6.13 | Calibre-Web Metadata cross site scripting | CVE-2021-25964 | +| V 0.6.13 | Name of Shelves are only visible to users who can access the corresponding shelf Thanks to @ibarrionuevo | | +| V 0.6.13 | JavaScript could get executed in the description field. Thanks to @ranjit-git and Hagai Wechsler (WhiteSource) | | +| V 0.6.13 | JavaScript could get executed in a custom column of type "comment" field | | +| V 0.6.13 | JavaScript could get executed after converting a book to another format with a title containing javascript code | | +| V 0.6.13 | JavaScript could get executed after converting a book to another format with a username containing javascript code | | +| V 0.6.13 | JavaScript could get executed in the description series, categories or publishers title | | +| V 0.6.13 | JavaScript could get executed in the shelf title | | +| V 0.6.13 | Login with the old session cookie after logout. Thanks to @ibarrionuevo | | +| V 0.6.14 | CSRF was possible. Thanks to @mik317 and Hagai Wechsler (WhiteSource) | CVE-2021-25965 | +| V 0.6.14 | Migrated some routes to POST-requests (CSRF protection). Thanks to @scara31 | CVE-2021-4164 | +| V 0.6.15 | Fix for "javascript:" script links in identifier. Thanks to @scara31 | CVE-2021-4170 | +| V 0.6.15 | Cross-Site Scripting vulnerability on uploaded cover file names. Thanks to @ibarrionuevo | | +| V 0.6.15 | Creating public shelfs is now denied if user is missing the edit public shelf right. Thanks to @ibarrionuevo | | +| V 0.6.15 | Changed error message in case of trying to delete a shelf unauthorized. Thanks to @ibarrionuevo | | +| V 0.6.16 | JavaScript could get executed on authors page. Thanks to @alicaz | CVE-2022-0352 | +| V 0.6.16 | Localhost can no longer be used to upload covers. Thanks to @scara31 | CVE-2022-0339 | +| V 0.6.16 | Another case where public shelfs could be created without permission is prevented. Thanks to @nhiephon | CVE-2022-0273 | +| V 0.6.16 | It's prevented to get the name of a private shelfs. Thanks to @nhiephon | CVE-2022-0405 | +| V 0.6.17 | The SSRF Protection can no longer be bypassed via an HTTP redirect. Thanks to @416e6e61 | CVE-2022-0767 | +| V 0.6.17 | The SSRF Protection can no longer be bypassed via 0.0.0.0 and it's ipv6 equivalent. Thanks to @r0hanSH | CVE-2022-0766 | +| V 0.6.18 | Possible SQL Injection is prevented in user table Thanks to Iman Sharafaldin (Forward Security) | CVE-2022-30765 | +| V 0.6.18 | The SSRF protection no longer can be bypassed by IPV6/IPV4 embedding. Thanks to @416e6e61 | CVE-2022-0939 | +| V 0.6.18 | The SSRF protection no longer can be bypassed to connect to other servers in the local network. Thanks to @michaellrowley | CVE-2022-0990 | +| V 0.6.20 | Credentials for emails are now stored encrypted | | +| V 0.6.20 | Login is rate limited | | +| V 0.6.20 | Passwordstrength can be forced | | +| V 0.6.21 | SMTP server credentials are no longer returned to client | | +| V 0.6.21 | Cross-site scripting (XSS) stored in href bypasses filter using data wrapper no longer possible | | +| V 0.6.21 | Cross-site scripting (XSS) is no longer possible via pathchooser | | +| V 0.6.21 | Error Handling at non existent rating, language, and user downloaded books was fixed | | +| V 0.6.22 | Upload mimetype is checked to prevent malicious file content in the books library | | +| V 0.6.22 | Cross-site scripting (XSS) stored in comments section is prevented better (switching from lxml to bleach for sanitizing strings) | | +| V 0.6.23 | Cookies are no longer stored for opds basic authentication and proxy authentication | | + + ## Statement regarding Log4j (CVE-2021-44228 and related) diff --git a/cps/__init__.py b/cps/__init__.py index d003ed5a..c4150942 100644 --- a/cps/__init__.py +++ b/cps/__init__.py @@ -35,7 +35,6 @@ from .reverseproxy import ReverseProxied from .server import WebServer from .dep_check import dependency_check from .updater import Updater -from .babel import babel, get_locale from . import config_sql from . import cache_buster from . import ub, db @@ -56,35 +55,41 @@ mimetypes.init() mimetypes.add_type('application/xhtml+xml', '.xhtml') mimetypes.add_type('application/epub+zip', '.epub') mimetypes.add_type('application/epub+zip', '.kepub') -mimetypes.add_type('text/xml', '.fb2') -mimetypes.add_type('application/octet-stream', '.mobi') +mimetypes.add_type('application/fb2+zip', '.fb2') +mimetypes.add_type('application/x-mobipocket-ebook', '.mobi') mimetypes.add_type('application/octet-stream', '.prc') -mimetypes.add_type('application/vnd.amazon.ebook', '.azw') -mimetypes.add_type('application/x-mobi8-ebook', '.azw3') -mimetypes.add_type('application/x-rar', '.cbr') -mimetypes.add_type('application/zip', '.cbz') +mimetypes.add_type('application/x-mobipocket-ebook', '.azw') +mimetypes.add_type('application/x-mobipocket-ebook', '.azw3') +mimetypes.add_type('application/x-cbr', '.cbr') +mimetypes.add_type('application/x-cbz', '.cbz') mimetypes.add_type('application/x-tar', '.cbt') mimetypes.add_type('application/x-7z-compressed', '.cb7') -mimetypes.add_type('image/vnd.djv', '.djv') -mimetypes.add_type('image/vnd.djv', '.djvu') +mimetypes.add_type('image/vnd.djvu', '.djv') +mimetypes.add_type('image/vnd.djvu', '.djvu') mimetypes.add_type('application/mpeg', '.mpeg') mimetypes.add_type('audio/mpeg', '.mp3') mimetypes.add_type('audio/x-m4a', '.m4a') mimetypes.add_type('audio/x-m4a', '.m4b') +mimetypes.add_type('audio/x-hx-aac-adts', '.aac') +mimetypes.add_type('audio/vnd.dolby.dd-raw', '.ac3') +mimetypes.add_type('video/x-ms-asf', '.asf') mimetypes.add_type('audio/ogg', '.ogg') mimetypes.add_type('application/ogg', '.oga') mimetypes.add_type('text/css', '.css') mimetypes.add_type('application/x-ms-reader', '.lit') -mimetypes.add_type('text/javascript; charset=UTF-8', '.js') +mimetypes.add_type('text/javascript', '.js') +mimetypes.add_type('text/rtf', '.rtf') log = logger.create() app = Flask(__name__) app.config.update( SESSION_COOKIE_HTTPONLY=True, - SESSION_COOKIE_SAMESITE='Strict', - REMEMBER_COOKIE_SAMESITE='Strict', # will be available in flask-login 0.5.1 earliest - WTF_CSRF_SSL_STRICT=False + SESSION_COOKIE_SAMESITE='Lax', + REMEMBER_COOKIE_SAMESITE='Strict', + WTF_CSRF_SSL_STRICT=False, + SESSION_COOKIE_NAME=os.environ.get('COOKIE_PREFIX', "") + "session", + REMEMBER_COOKIE_NAME=os.environ.get('COOKIE_PREFIX', "") + "remember_token" ) lm = MyLoginManager() @@ -98,7 +103,7 @@ if wtf_present: else: csrf = None -calibre_db = db.CalibreDB() +calibre_db = db.CalibreDB(app) web_server = WebServer() @@ -142,9 +147,7 @@ def create_app(): lm.anonymous_user = ub.Anonymous lm.session_protection = 'strong' if config.config_session == 1 else "basic" - db.CalibreDB.update_config(config) - db.CalibreDB.setup_db(config.config_calibre_dir, cli_param.settings_path) - calibre_db.init_db() + db.CalibreDB.update_config(config, config.config_calibre_dir, cli_param.settings_path) updater_thread.init_updater(config, web_server) # Perform dry run of updater and exit afterward @@ -177,6 +180,7 @@ def create_app(): app.secret_key = os.getenv('SECRET_KEY', config_sql.get_flask_session_key(ub.session)) web_server.init_app(app, config) + from .cw_babel import babel, get_locale if hasattr(babel, "localeselector"): babel.init_app(app) babel.localeselector(get_locale) diff --git a/cps/about.py b/cps/about.py index 355978c3..d76fc510 100644 --- a/cps/about.py +++ b/cps/about.py @@ -23,10 +23,10 @@ import sys import platform import sqlite3 +from importlib.metadata import metadata from collections import OrderedDict import flask -import jinja2 from flask_babel import gettext as _ from . import db, calibre_db, converter, uploader, constants, dep_check @@ -41,17 +41,18 @@ req = dep_check.load_dependencies(False) opt = dep_check.load_dependencies(True) for i in (req + opt): modules[i[1]] = i[0] -modules['Jinja2'] = jinja2.__version__ -modules['pySqlite'] = sqlite3.version +modules['Jinja2'] = metadata("jinja2")["Version"] +if sys.version_info < (3, 12): + modules['pySqlite'] = sqlite3.version modules['SQLite'] = sqlite3.sqlite_version sorted_modules = OrderedDict((sorted(modules.items(), key=lambda x: x[0].casefold()))) def collect_stats(): if constants.NIGHTLY_VERSION[0] == "$Format:%H$": - calibre_web_version = constants.STABLE_VERSION['version'].replace("b", " Beta") + calibre_web_version = constants.STABLE_VERSION.replace("b", " Beta") else: - calibre_web_version = (constants.STABLE_VERSION['version'].replace("b", " Beta") + ' - ' + calibre_web_version = (constants.STABLE_VERSION.replace("b", " Beta") + ' - ' + constants.NIGHTLY_VERSION[0].replace('%', '%%') + ' - ' + constants.NIGHTLY_VERSION[1].replace('%', '%%')) diff --git a/cps/admin.py b/cps/admin.py index 5f481fa1..5451080a 100644 --- a/cps/admin.py +++ b/cps/admin.py @@ -32,7 +32,8 @@ from datetime import time as datetime_time from functools import wraps from urllib.parse import urlparse -from flask import Blueprint, flash, redirect, url_for, abort, request, make_response, send_from_directory, g, Response +from flask import Blueprint, flash, redirect, url_for, abort, request, make_response, \ + send_from_directory, g, jsonify from markupsafe import Markup from .cw_login import current_user from flask_babel import gettext as _ @@ -52,8 +53,9 @@ from .gdriveutils import is_gdrive_ready, gdrive_support from .render_template import render_title_template, get_sidebar_config from .services.worker import WorkerThread from .usermanagement import user_login_required -from .babel import get_available_translations, get_available_locale, get_user_locale_language +from .cw_babel import get_available_translations, get_available_locale, get_user_locale_language from . import debug_info +from .string_helper import strip_whitespaces log = logger.create() @@ -117,7 +119,7 @@ def before_request(): g.allow_upload = config.config_uploading g.current_theme = config.config_theme g.config_authors_max = config.config_authors_max - if '/static/' not in request.path and not config.db_configured and \ + if ('/static/' not in request.path and not config.db_configured and request.endpoint not in ('admin.ajax_db_config', 'admin.simulatedbchange', 'admin.db_configuration', @@ -125,7 +127,7 @@ def before_request(): 'web.login_post', 'web.logout', 'admin.load_dialogtexts', - 'admin.ajax_pathchooser'): + 'admin.ajax_pathchooser')): return redirect(url_for('admin.db_configuration')) @@ -143,7 +145,6 @@ def shutdown(): show_text = {} if task in (0, 1): # valid commandos received # close all database connections - calibre_db.dispose() ub.dispose() if task == 0: @@ -305,7 +306,15 @@ def edit_user_table(): .group_by(text('books_tags_link.tag')) \ .order_by(db.Tags.name).all() if config.config_restricted_column: - custom_values = calibre_db.session.query(db.cc_classes[config.config_restricted_column]).all() + try: + custom_values = calibre_db.session.query(db.cc_classes[config.config_restricted_column]).all() + except (KeyError, AttributeError, IndexError): + custom_values = [] + log.error("Custom Column No.{} does not exist in calibre database".format( + config.config_restricted_column)) + flash(_("Custom Column No.%(column)d does not exist in calibre database", + column=config.config_restricted_column), + category="error") else: custom_values = [] if not config.config_anonbrowse: @@ -370,10 +379,7 @@ def list_users(): user.default = get_user_locale_language(user.default_language) table_entries = {'totalNotFiltered': total_count, 'total': filtered_count, "rows": users} - js_list = json.dumps(table_entries, cls=db.AlchemyEncoder) - response = make_response(js_list) - response.headers["Content-Type"] = "application/json; charset=utf-8" - return response + return make_response(json.dumps(table_entries, cls=db.AlchemyEncoder)) @admi.route("/ajax/deleteuser", methods=['POST']) @@ -392,7 +398,7 @@ def delete_user(): success = list() if not users: log.error("User not found") - return Response(json.dumps({'type': "danger", 'message': _("User not found")}), mimetype='application/json') + return make_response(jsonify(type="danger", message=_("User not found"))) for user in users: try: message = _delete_user(user) @@ -408,7 +414,7 @@ def delete_user(): log.info("Users {} deleted".format(user_ids)) success = [{'type': "success", 'message': _("{} users deleted successfully").format(count)}] success.extend(errors) - return Response(json.dumps(success), mimetype='application/json') + return make_response(jsonify(success)) @admi.route("/ajax/getlocale") @@ -420,7 +426,7 @@ def table_get_locale(): current_locale = get_locale() for loc in locale: ret.append({'value': str(loc), 'text': loc.get_language_name(current_locale)}) - return json.dumps(ret) + return json.dumps(sorted(ret, key=lambda x: x['text'])) @admi.route("/ajax/getdefaultlanguage") @@ -432,7 +438,7 @@ def table_get_default_lang(): ret.append({'value': 'all', 'text': _('Show All')}) for lang in languages: ret.append({'value': lang.lang_code, 'text': lang.name}) - return json.dumps(ret) + return json.dumps(sorted(ret, key=lambda x: x['text'])) @admi.route("/ajax/editlistusers/", methods=['POST']) @@ -463,9 +469,9 @@ def edit_list_user(param): if 'value[]' in vals: setattr(user, param, prepare_tags(user, vals['action'][0], param, vals['value[]'])) else: - setattr(user, param, vals['value'].strip()) + setattr(user, param, strip_whitespaces(vals['value'])) else: - vals['value'] = vals['value'].strip() + vals['value'] = strip_whitespaces(vals['value']) if param == 'name': if user.name == "Guest": raise Exception(_("Guest Name can't be changed")) @@ -490,10 +496,10 @@ def edit_list_user(param): if not ub.session.query(ub.User). \ filter(ub.User.role.op('&')(constants.ROLE_ADMIN) == constants.ROLE_ADMIN, ub.User.id != user.id).count(): - return Response( - json.dumps([{'type': "danger", + return make_response( + jsonify([{'type': "danger", 'message': _("No admin user remaining, can't remove admin role", - nick=user.name)}]), mimetype='application/json') + nick=user.name)}])) user.role &= ~value else: raise Exception(_("Value has to be true or false")) @@ -566,7 +572,7 @@ def update_view_configuration(): _config_string(to_save, "config_calibre_web_title") _config_string(to_save, "config_columns_to_ignore") if _config_string(to_save, "config_title_regex"): - calibre_db.update_title_sort(config) + calibre_db.create_functions(config) if not check_valid_read_column(to_save.get("config_read_column", "0")): flash(_("Invalid Read Column"), category="error") @@ -690,7 +696,7 @@ def delete_domain(): def list_domain(allow): answer = ub.session.query(ub.Registration).filter(ub.Registration.allow == allow).all() json_dumps = json.dumps([{"domain": r.domain.replace('%', '*').replace('_', '?'), "id": r.id} for r in answer]) - js = json.dumps(json_dumps.replace('"', "'")).lstrip('"').strip('"') + js = json.dumps(json_dumps.replace('"', "'")).strip('"') response = make_response(js.replace("'", '"')) response.headers["Content-Type"] = "application/json; charset=utf-8" return response @@ -939,7 +945,7 @@ def do_full_kobo_sync(userid): count = ub.session.query(ub.KoboSyncedBooks).filter(userid == ub.KoboSyncedBooks.user_id).delete() message = _("{} sync entries deleted").format(count) ub.session_commit(message) - return Response(json.dumps([{"type": "success", "message": message}]), mimetype='application/json') + return make_response(jsonify(type="success", message=message)) def check_valid_read_column(column): @@ -981,8 +987,14 @@ def prepare_tags(user, action, tags_name, id_list): raise Exception(_("Tag not found")) new_tags_list = [x.name for x in tags] else: - tags = calibre_db.session.query(db.cc_classes[config.config_restricted_column]) \ - .filter(db.cc_classes[config.config_restricted_column].id.in_(id_list)).all() + try: + tags = calibre_db.session.query(db.cc_classes[config.config_restricted_column]) \ + .filter(db.cc_classes[config.config_restricted_column].id.in_(id_list)).all() + except (KeyError, AttributeError, IndexError): + log.error("Custom Column No.{} does not exist in calibre database".format( + config.config_restricted_column)) + raise Exception(_("Custom Column No.%(column)d does not exist in calibre database", + column=config.config_restricted_column)) new_tags_list = [x.value for x in tags] saved_tags_list = user.__dict__[tags_name].split(",") if len(user.__dict__[tags_name]) else [] if action == "remove": @@ -1100,7 +1112,7 @@ def _config_checkbox_int(to_save, x): def _config_string(to_save, x): - return config.set_from_dictionary(to_save, x, lambda y: y.strip().strip(u'\u200B\u200C\u200D\ufeff') if y else y) + return config.set_from_dictionary(to_save, x, lambda y: strip_whitespaces(y) if y else y) def _configuration_gdrive_helper(to_save): @@ -1250,7 +1262,7 @@ def _configuration_ldap_helper(to_save): @admin_required def simulatedbchange(): db_change, db_valid = _db_simulate_change() - return Response(json.dumps({"change": db_change, "valid": db_valid}), mimetype='application/json') + return make_response(jsonify(change=db_change, valid=db_valid)) @admi.route("/admin/user/new", methods=["GET", "POST"]) @@ -1311,9 +1323,9 @@ def update_mailsettings(): if to_save.get("mail_password_e", ""): _config_string(to_save, "mail_password_e") _config_int(to_save, "mail_size", lambda y: int(y) * 1024 * 1024) - config.mail_server = to_save.get('mail_server', "").strip() - config.mail_from = to_save.get('mail_from', "").strip() - config.mail_login = to_save.get('mail_login', "").strip() + config.mail_server = strip_whitespaces(to_save.get('mail_server', "")) + config.mail_from = strip_whitespaces(to_save.get('mail_from', "")) + config.mail_login = strip_whitespaces(to_save.get('mail_login', "")) try: config.save() except (OperationalError, InvalidRequestError) as e: @@ -1678,10 +1690,10 @@ def cancel_task(): def _db_simulate_change(): param = request.form.to_dict() to_save = dict() - to_save['config_calibre_dir'] = re.sub(r'[\\/]metadata\.db$', + to_save['config_calibre_dir'] = strip_whitespaces(re.sub(r'[\\/]metadata\.db$', '', param['config_calibre_dir'], - flags=re.IGNORECASE).strip() + flags=re.IGNORECASE)) db_valid, db_change = calibre_db.check_valid_db(to_save["config_calibre_dir"], ub.app_DB_path, config.config_calibre_uuid) @@ -1715,14 +1727,19 @@ def _db_configuration_update_helper(): db_change = True except Exception as ex: return _db_configuration_result('{}'.format(ex), gdrive_error) - - if db_change or not db_valid or not config.db_configured \ - or config.config_calibre_dir != to_save["config_calibre_dir"]: + config.config_calibre_split = to_save.get('config_calibre_split', 0) == "on" + if config.config_calibre_split: + split_dir = to_save.get("config_calibre_split_dir") + if not os.path.exists(split_dir): + return _db_configuration_result(_("Books path not valid"), gdrive_error) + else: + _config_string(to_save, "config_calibre_split_dir") + if (db_change or not db_valid or not config.db_configured + or config.config_calibre_dir != to_save["config_calibre_dir"]): if not os.path.exists(metadata_db) or not to_save['config_calibre_dir']: return _db_configuration_result(_('DB Location is not Valid, Please Enter Correct Path'), gdrive_error) else: calibre_db.setup_db(to_save['config_calibre_dir'], ub.app_DB_path) - config.store_calibre_uuid(calibre_db, db.Library_Id) # if db changed -> delete shelfs, delete download books, delete read books, kobo sync... if db_change: log.info("Calibre Database changed, all Calibre-Web info related to old Database gets deleted") @@ -1736,13 +1753,20 @@ def _db_configuration_update_helper(): ub.session.query(ub.KoboSyncedBooks).delete() helper.delete_thumbnail_cache() ub.session_commit() + # deleted visibilities based on custom column and tags + config.config_restricted_column = 0 + config.config_denied_tags = "" + config.config_allowed_tags = "" + config.config_columns_to_ignore = "" + config.config_denied_column_value = "" + config.config_allowed_column_value = "" + config.config_read_column = 0 _config_string(to_save, "config_calibre_dir") - calibre_db.update_config(config) + calibre_db.update_config(config, config.config_calibre_dir, ub.app_DB_path) + config.store_calibre_uuid(calibre_db, db.Library_Id) if not os.access(os.path.join(config.config_calibre_dir, "metadata.db"), os.W_OK): flash(_("DB is not Writeable"), category="warning") - _config_string(to_save, "config_calibre_split_dir") - config.config_calibre_split = to_save.get('config_calibre_split', 0) == "on" - calibre_db.update_config(config) + calibre_db.update_config(config, config.config_calibre_dir, ub.app_DB_path) config.save() return _db_configuration_result(None, gdrive_error) @@ -1775,9 +1799,8 @@ def _configuration_update_helper(): if "config_upload_formats" in to_save: to_save["config_upload_formats"] = ','.join( - helper.uniq([x.lstrip().rstrip().lower() for x in to_save["config_upload_formats"].split(',')])) + helper.uniq([x.strip().lower() for x in to_save["config_upload_formats"].split(',')])) _config_string(to_save, "config_upload_formats") - # constants.EXTENSIONS_UPLOAD = config.config_upload_formats.split(',') _config_string(to_save, "config_calibre") _config_string(to_save, "config_binariesdir") @@ -1871,7 +1894,7 @@ def _configuration_result(error_flash=None, reboot=False): resp['result'] = [{'type': "success", 'message': _("Calibre-Web configuration updated")}] resp['reboot'] = reboot resp['config_upload'] = config.config_upload_formats - return Response(json.dumps(resp), mimetype='application/json') + return make_response(jsonify(resp)) def _db_configuration_result(error_flash=None, gdrive_error=None): @@ -2076,7 +2099,7 @@ def _handle_edit_user(to_save, content, languages, translations, kobo_support): def extract_user_data_from_field(user, field): - match = re.search(field + r"=([@\.\d\s\w-]+)", user, re.IGNORECASE | re.UNICODE) + match = re.search(field + r"=(.*?)($|(?. + +import mutagen +import base64 +from . import cover, logger + +from cps.constants import BookMeta + +log = logger.create() + +def get_audio_file_info(tmp_file_path, original_file_extension, original_file_name, no_cover_processing): + tmp_cover_name = None + audio_file = mutagen.File(tmp_file_path) + comments = None + if original_file_extension in [".mp3", ".wav", ".aiff"]: + cover_data = list() + for key, val in audio_file.tags.items(): + if key.startswith("APIC:"): + cover_data.append(val) + if key.startswith("COMM:"): + comments = val.text[0] + title = audio_file.tags.get('TIT2').text[0] if "TIT2" in audio_file.tags else None + author = audio_file.tags.get('TPE1').text[0] if "TPE1" in audio_file.tags else None + if author is None: + author = audio_file.tags.get('TPE2').text[0] if "TPE2" in audio_file.tags else None + tags = audio_file.tags.get('TCON').text[0] if "TCON" in audio_file.tags else None # Genre + series = audio_file.tags.get('TALB').text[0] if "TALB" in audio_file.tags else None# Album + series_id = audio_file.tags.get('TRCK').text[0] if "TRCK" in audio_file.tags else None # track no. + publisher = audio_file.tags.get('TPUB').text[0] if "TPUB" in audio_file.tags else None + pubdate = str(audio_file.tags.get('TDRL').text[0]) if "TDRL" in audio_file.tags else None + if not pubdate: + pubdate = str(audio_file.tags.get('TDRC').text[0]) if "TDRC" in audio_file.tags else None + if not pubdate: + pubdate = str(audio_file.tags.get('TDOR').text[0]) if "TDOR" in audio_file.tags else None + if cover_data and not no_cover_processing: + cover_info = cover_data[0] + for dat in cover_data: + if dat.type == mutagen.id3.PictureType.COVER_FRONT: + cover_info = dat + break + tmp_cover_name = cover.cover_processing(tmp_file_path, cover_info.data, "." + cover_info.mime[-3:]) + elif original_file_extension in [".ogg", ".flac", ".opus", ".ogv"]: + title = audio_file.tags.get('TITLE')[0] if "TITLE" in audio_file else None + author = audio_file.tags.get('ARTIST')[0] if "ARTIST" in audio_file else None + comments = audio_file.tags.get('COMMENTS')[0] if "COMMENTS" in audio_file else None + tags = audio_file.tags.get('GENRE')[0] if "GENRE" in audio_file else None # Genre + series = audio_file.tags.get('ALBUM')[0] if "ALBUM" in audio_file else None + series_id = audio_file.tags.get('TRACKNUMBER')[0] if "TRACKNUMBER" in audio_file else None + publisher = audio_file.tags.get('LABEL')[0] if "LABEL" in audio_file else None + pubdate = audio_file.tags.get('DATE')[0] if "DATE" in audio_file else None + cover_data = audio_file.tags.get('METADATA_BLOCK_PICTURE') + if not no_cover_processing: + if cover_data: + cover_info = mutagen.flac.Picture(base64.b64decode(cover_data[0])) + tmp_cover_name = cover.cover_processing(tmp_file_path, cover_info.data, "." + cover_info.mime[-3:]) + if hasattr(audio_file, "pictures"): + cover_info = audio_file.pictures[0] + for dat in audio_file.pictures: + if dat.type == mutagen.id3.PictureType.COVER_FRONT: + cover_info = dat + break + tmp_cover_name = cover.cover_processing(tmp_file_path, cover_info.data, "." + cover_info.mime[-3:]) + elif original_file_extension in [".aac"]: + title = audio_file.tags.get('Title').value if "Title" in audio_file else None + author = audio_file.tags.get('Artist').value if "Artist" in audio_file else None + comments = audio_file.tags.get('Comment').value if "Comment" in audio_file else None + tags = audio_file.tags.get('Genre').value if "Genre" in audio_file else None + series = audio_file.tags.get('Album').value if "Album" in audio_file else None + series_id = audio_file.tags.get('Track').value if "Track" in audio_file else None + publisher = audio_file.tags.get('Label').value if "Label" in audio_file else None + pubdate = audio_file.tags.get('Year').value if "Year" in audio_file else None + cover_data = audio_file.tags['Cover Art (Front)'] + if cover_data and not no_cover_processing: + tmp_cover_name = tmp_file_path + '.jpg' + with open(tmp_cover_name, "wb") as cover_file: + cover_file.write(cover_data.value.split(b"\x00",1)[1]) + elif original_file_extension in [".asf"]: + title = audio_file.tags.get('Title')[0].value if "Title" in audio_file else None + author = audio_file.tags.get('Artist')[0].value if "Artist" in audio_file else None + comments = audio_file.tags.get('Comments')[0].value if "Comments" in audio_file else None + tags = audio_file.tags.get('Genre')[0].value if "Genre" in audio_file else None + series = audio_file.tags.get('Album')[0].value if "Album" in audio_file else None + series_id = audio_file.tags.get('Track')[0].value if "Track" in audio_file else None + publisher = audio_file.tags.get('Label')[0].value if "Label" in audio_file else None + pubdate = audio_file.tags.get('Year')[0].value if "Year" in audio_file else None + cover_data = audio_file.tags.get('WM/Picture', None) + if cover_data and not no_cover_processing: + tmp_cover_name = tmp_file_path + '.jpg' + with open(tmp_cover_name, "wb") as cover_file: + cover_file.write(cover_data[0].value) + elif original_file_extension in [".mp4", ".m4a", ".m4b"]: + title = audio_file.tags.get('©nam')[0] if "©nam" in audio_file.tags else None + author = audio_file.tags.get('©ART')[0] if "©ART" in audio_file.tags else None + comments = audio_file.tags.get('©cmt')[0] if "©cmt" in audio_file.tags else None + tags = audio_file.tags.get('©gen')[0] if "©gen" in audio_file.tags else None + series = audio_file.tags.get('©alb')[0] if "©alb" in audio_file.tags else None + series_id = str(audio_file.tags.get('trkn')[0][0]) if "trkn" in audio_file.tags else None + publisher = "" + pubdate = audio_file.tags.get('©day')[0] if "©day" in audio_file.tags else None + cover_data = audio_file.tags.get('covr', None) + if cover_data and not no_cover_processing: + cover_type = None + for c in cover_data: + if c.imageformat == mutagen.mp4.AtomDataType.JPEG: + cover_type =".jpg" + cover_bin = c + break + elif c.imageformat == mutagen.mp4.AtomDataType.PNG: + cover_type = ".png" + cover_bin = c + break + if cover_type: + tmp_cover_name = cover.cover_processing(tmp_file_path, cover_bin, cover_type) + else: + logger.error("Unknown covertype in file {} ".format(original_file_name)) + + return BookMeta( + file_path=tmp_file_path, + extension=original_file_extension, + title=title or original_file_name , + author="Unknown" if author is None else author, + cover=tmp_cover_name, + description="" if comments is None else comments, + tags="" if tags is None else tags, + series="" if series is None else series, + series_id="1" if series_id is None else series_id.split("/")[0], + languages="", + publisher= "" if publisher is None else publisher, + pubdate="" if pubdate is None else pubdate, + identifiers=[], + ) diff --git a/cps/basic.py b/cps/basic.py new file mode 100644 index 00000000..b0121010 --- /dev/null +++ b/cps/basic.py @@ -0,0 +1,90 @@ +# This file is part of the Calibre-Web (https://github.com/janeczku/calibre-web) +# Copyright (C) 2018-2019 OzzieIsaacs, cervinko, jkrehm, bodybybuddha, ok11, +# andy29485, idalin, Kyosfonica, wuqi, Kennyl, lemmsh, +# falgh1, grunjol, csitko, ytils, xybydy, trasba, vrabe, +# ruben-herold, marblepebble, JackED42, SiphonSquirrel, +# apetresc, nanu-c, mutschler, carderne +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . + + +from cps.pagination import Pagination +from flask import Blueprint +from flask_babel import gettext as _ +from flask_babel import get_locale +from flask import request, redirect, url_for + +from . import logger, isoLanguages +from . import db, config +from . import calibre_db +from .usermanagement import login_required_if_no_ano +from .render_template import render_title_template +from .web import get_sort_function + +try: + from natsort import natsorted as sort +except ImportError: + sort = sorted # Just use regular sort then, may cause issues with badly named pages in cbz/cbr files + +basic = Blueprint('basic', __name__) + +log = logger.create() + + +@basic.route("/basic", methods=["GET"]) +@login_required_if_no_ano +def index(): + term = request.args.get("query", "") # default to showing all books + limit = 15 + page = int(request.args.get("page") or 1) + off = (page - 1) * limit + order = get_sort_function("stored", "search") + join = db.books_series_link, db.Books.id == db.books_series_link.c.book, db.Series + entries, result_count, pagination = calibre_db.get_search_results(term, + config, + off, + order, + limit, + *join) + return render_title_template('basic_index.html', + searchterm=term, + pagination=pagination, + query=term, + adv_searchterm=term, + entries=entries, + result_count=result_count, + title=_("Search"), + page="search", + order=order[1]) + + +@basic.route("/basic_book/") +@login_required_if_no_ano +def show_book(book_id): + entries = calibre_db.get_book_read_archived(book_id, config.config_read_column, allow_show_archived=True) + if entries: + entry = entries[0] + for lang_index in range(0, len(entry.languages)): + entry.languages[lang_index].language_name = isoLanguages.get_language_name(get_locale(), entry.languages[ + lang_index].lang_code) + entry.ordered_authors = calibre_db.order_authors([entry]) + + return render_title_template('basic_detail.html', + entry=entry, + is_xhr=request.headers.get('X-Requested-With') == 'XMLHttpRequest', + title=entry.title, + page="book") + else: + log.debug("Selected book is unavailable. File does not exist or is not accessible") + return redirect(url_for("basic.index")) diff --git a/cps/cli.py b/cps/cli.py index 5bf289b8..3f21a17e 100644 --- a/cps/cli.py +++ b/cps/cli.py @@ -29,8 +29,8 @@ from .constants import DEFAULT_SETTINGS_FILE, DEFAULT_GDRIVE_FILE def version_info(): if _NIGHTLY_VERSION[1].startswith('$Format'): - return "Calibre-Web version: %s - unknown git-clone" % _STABLE_VERSION['version'].replace("b", " Beta") - return "Calibre-Web version: %s -%s" % (_STABLE_VERSION['version'].replace("b", " Beta"), _NIGHTLY_VERSION[1]) + return "Calibre-Web version: %s - unknown git-clone" % _STABLE_VERSION.replace("b", " Beta") + return "Calibre-Web version: %s -%s" % (_STABLE_VERSION.replace("b", " Beta"), _NIGHTLY_VERSION[1]) class CliParameter(object): diff --git a/cps/comic.py b/cps/comic.py index 27c86c9a..271b1b45 100644 --- a/cps/comic.py +++ b/cps/comic.py @@ -90,7 +90,7 @@ def _extract_cover_from_archive(original_file_extension, tmp_file_name, rar_exec if len(ext) > 1: extension = ext[1].lower() if extension in cover.COVER_EXTENSIONS: - cover_data = cf.read([name]) + cover_data = cf.read(name) break except Exception as ex: log.error('Rarfile failed with error: {}'.format(ex)) @@ -109,13 +109,13 @@ def _extract_cover_from_archive(original_file_extension, tmp_file_name, rar_exec return cover_data, extension -def _extract_cover(tmp_file_name, original_file_extension, rar_executable): +def _extract_cover(tmp_file_path, original_file_extension, rar_executable): cover_data = extension = None if use_comic_meta: try: - archive = ComicArchive(tmp_file_name, rar_exe_path=rar_executable) + archive = ComicArchive(tmp_file_path, rar_exe_path=rar_executable) except TypeError: - archive = ComicArchive(tmp_file_name) + archive = ComicArchive(tmp_file_path) name_list = archive.getPageNameList if hasattr(archive, "getPageNameList") else archive.get_page_name_list for index, name in enumerate(name_list()): ext = os.path.splitext(name) @@ -126,11 +126,11 @@ def _extract_cover(tmp_file_name, original_file_extension, rar_executable): cover_data = get_page(index) break else: - cover_data, extension = _extract_cover_from_archive(original_file_extension, tmp_file_name, rar_executable) - return cover.cover_processing(tmp_file_name, cover_data, extension) + cover_data, extension = _extract_cover_from_archive(original_file_extension, tmp_file_path, rar_executable) + return cover.cover_processing(tmp_file_path, cover_data, extension) -def get_comic_info(tmp_file_path, original_file_name, original_file_extension, rar_executable): +def get_comic_info(tmp_file_path, original_file_name, original_file_extension, rar_executable, no_cover_processing): if use_comic_meta: try: archive = ComicArchive(tmp_file_path, rar_exe_path=rar_executable) @@ -155,14 +155,17 @@ def get_comic_info(tmp_file_path, original_file_name, original_file_extension, r lang = loaded_metadata.language or "" loaded_metadata.language = isoLanguages.get_lang3(lang) - + if not no_cover_processing: + cover_file = _extract_cover(tmp_file_path, original_file_extension, rar_executable) + else: + cover_file = None return BookMeta( file_path=tmp_file_path, extension=original_file_extension, title=loaded_metadata.title or original_file_name, author=" & ".join([credit["person"] for credit in loaded_metadata.credits if credit["role"] == "Writer"]) or 'Unknown', - cover=_extract_cover(tmp_file_path, original_file_extension, rar_executable), + cover=cover_file, description=loaded_metadata.comments or "", tags="", series=loaded_metadata.series or "", @@ -171,13 +174,17 @@ def get_comic_info(tmp_file_path, original_file_name, original_file_extension, r publisher="", pubdate="", identifiers=[]) + if not no_cover_processing: + cover_file = _extract_cover(tmp_file_path, original_file_extension, rar_executable) + else: + cover_file = None return BookMeta( file_path=tmp_file_path, extension=original_file_extension, title=original_file_name, author='Unknown', - cover=_extract_cover(tmp_file_path, original_file_extension, rar_executable), + cover=cover_file, description="", tags="", series="", diff --git a/cps/config_sql.py b/cps/config_sql.py index 044c12b5..7e7d033d 100644 --- a/cps/config_sql.py +++ b/cps/config_sql.py @@ -35,7 +35,7 @@ except ImportError: from . import constants, logger from .subproc_wrapper import process_wait - +from .string_helper import strip_whitespaces log = logger.create() _Base = declarative_base() @@ -182,26 +182,6 @@ class _Settings(_Base): class ConfigSQL(object): # pylint: disable=no-member def __init__(self): - '''self.config_calibre_uuid = None - self.config_calibre_split_dir = None - self.dirty = None - self.config_logfile = None - self.config_upload_formats = None - self.mail_gmail_token = None - self.mail_server_type = None - self.mail_server = None - self.config_log_level = None - self.config_allowed_column_value = None - self.config_denied_column_value = None - self.config_allowed_tags = None - self.config_denied_tags = None - self.config_default_show = None - self.config_default_role = None - self.config_keyfile = None - self.config_certfile = None - self.config_rarfile_location = None - self.config_kepubifypath = None - self.config_binariesdir = None''' self.__dict__["dirty"] = list() def init_config(self, session, secret_key, cli): @@ -288,19 +268,19 @@ class ConfigSQL(object): def list_denied_tags(self): mct = self.config_denied_tags or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def list_allowed_tags(self): mct = self.config_allowed_tags or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def list_denied_column_values(self): mct = self.config_denied_column_value or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def list_allowed_column_values(self): mct = self.config_allowed_column_value or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def get_log_level(self): return logger.get_level_name(self.config_log_level) @@ -372,7 +352,7 @@ class ConfigSQL(object): db_file = os.path.join(self.config_calibre_dir, 'metadata.db') have_metadata_db = os.path.isfile(db_file) self.db_configured = have_metadata_db - # constants.EXTENSIONS_UPLOAD = [x.lstrip().rstrip().lower() for x in self.config_upload_formats.split(',')] + from . import cli_param if os.environ.get('FLASK_DEBUG'): logfile = logger.setup(logger.LOG_TO_STDOUT, logger.logging.DEBUG) @@ -425,11 +405,13 @@ class ConfigSQL(object): return self.config_calibre_split_dir if self.config_calibre_split_dir else self.config_calibre_dir def store_calibre_uuid(self, calibre_db, Library_table): + from . import app try: - calibre_uuid = calibre_db.session.query(Library_table).one_or_none() - if self.config_calibre_uuid != calibre_uuid.uuid: - self.config_calibre_uuid = calibre_uuid.uuid - self.save() + with app.app_context(): + calibre_uuid = calibre_db.session.query(Library_table).one_or_none() + if self.config_calibre_uuid != calibre_uuid.uuid: + self.config_calibre_uuid = calibre_uuid.uuid + self.save() except AttributeError: pass @@ -503,6 +485,8 @@ def autodetect_calibre_binaries(): "C:\\program files(x86)\\calibre\\", "C:\\program files(x86)\\calibre2\\", "C:\\program files\\calibre2\\"] + elif sys.platform.startswith("freebsd"): + calibre_path = ["/usr/local/bin/"] else: calibre_path = ["/opt/calibre/"] for element in calibre_path: @@ -533,6 +517,8 @@ def autodetect_unrar_binary(): if sys.platform == "win32": calibre_path = ["C:\\program files\\WinRar\\unRAR.exe", "C:\\program files(x86)\\WinRar\\unRAR.exe"] + elif sys.platform.startswith("freebsd"): + calibre_path = ["/usr/local/bin/unrar"] else: calibre_path = ["/usr/bin/unrar"] for element in calibre_path: @@ -545,6 +531,8 @@ def autodetect_kepubify_binary(): if sys.platform == "win32": calibre_path = ["C:\\program files\\kepubify\\kepubify-windows-64Bit.exe", "C:\\program files(x86)\\kepubify\\kepubify-windows-64Bit.exe"] + elif sys.platform.startswith("freebsd"): + calibre_path = ["/usr/local/bin/kepubify"] else: calibre_path = ["/opt/kepubify/kepubify-linux-64bit", "/opt/kepubify/kepubify-linux-32bit"] for element in calibre_path: diff --git a/cps/constants.py b/cps/constants.py index 13146845..48b5a106 100644 --- a/cps/constants.py +++ b/cps/constants.py @@ -19,9 +19,6 @@ import sys import os from collections import namedtuple -from sqlalchemy import __version__ as sql_version - -sqlalchemy_version2 = ([int(x) for x in sql_version.split('.')] >= [2, 0, 0]) # APP_MODE - production, development, or test APP_MODE = os.environ.get('APP_MODE', 'production') @@ -175,7 +172,7 @@ BookMeta = namedtuple('BookMeta', 'file_path, extension, title, author, cover, d 'series_id, languages, publisher, pubdate, identifiers') # python build process likes to have x.y.zbw -> b for beta and w a counting number -STABLE_VERSION = {'version': '0.6.23b'} +STABLE_VERSION = '0.6.25b' NIGHTLY_VERSION = dict() NIGHTLY_VERSION[0] = '$Format:%H$' @@ -193,7 +190,7 @@ THUMBNAIL_TYPE_AUTHOR = 3 COVER_THUMBNAIL_ORIGINAL = 0 COVER_THUMBNAIL_SMALL = 1 COVER_THUMBNAIL_MEDIUM = 2 -COVER_THUMBNAIL_LARGE = 3 +COVER_THUMBNAIL_LARGE = 4 # clean-up the module namespace del sys, os, namedtuple diff --git a/cps/cover.py b/cps/cover.py index 5dd29534..6c42d73e 100644 --- a/cps/cover.py +++ b/cps/cover.py @@ -29,13 +29,14 @@ NO_JPEG_EXTENSIONS = ['.png', '.webp', '.bmp'] COVER_EXTENSIONS = ['.png', '.webp', '.bmp', '.jpg', '.jpeg'] -def cover_processing(tmp_file_name, img, extension): - tmp_cover_name = os.path.join(os.path.dirname(tmp_file_name), 'cover.jpg') +def cover_processing(tmp_file_path, img, extension): + # tmp_cover_name = os.path.join(os.path.dirname(tmp_file_name), 'cover.jpg') + tmp_cover_name = tmp_file_path + '.jpg' if extension in NO_JPEG_EXTENSIONS: if use_IM: with Image(blob=img) as imgc: imgc.format = 'jpeg' - imgc.transform_colorspace('rgb') + imgc.transform_colorspace('srgb') imgc.save(filename=tmp_cover_name) return tmp_cover_name else: diff --git a/cps/cw_advocate/__init__.py b/cps/cw_advocate/__init__.py new file mode 100644 index 00000000..58407b7b --- /dev/null +++ b/cps/cw_advocate/__init__.py @@ -0,0 +1,22 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + + +from .adapters import ValidatingHTTPAdapter +from .api import * +from .addrvalidator import AddrValidator +from .exceptions import UnacceptableAddressException diff --git a/cps/cw_advocate/adapters.py b/cps/cw_advocate/adapters.py new file mode 100644 index 00000000..b15a141d --- /dev/null +++ b/cps/cw_advocate/adapters.py @@ -0,0 +1,48 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +from requests.adapters import HTTPAdapter, DEFAULT_POOLBLOCK + +from .addrvalidator import AddrValidator +from .exceptions import ProxyDisabledException +from .poolmanager import ValidatingPoolManager + + +class ValidatingHTTPAdapter(HTTPAdapter): + __attrs__ = HTTPAdapter.__attrs__ + ['_validator'] + + def __init__(self, *args, **kwargs): + self._validator = kwargs.pop('validator', None) + if not self._validator: + self._validator = AddrValidator() + super().__init__(*args, **kwargs) + + def init_poolmanager(self, connections, maxsize, block=DEFAULT_POOLBLOCK, + **pool_kwargs): + self._pool_connections = connections + self._pool_maxsize = maxsize + self._pool_block = block + self.poolmanager = ValidatingPoolManager( + num_pools=connections, + maxsize=maxsize, + block=block, + validator=self._validator, + **pool_kwargs + ) + + def proxy_manager_for(self, proxy, **proxy_kwargs): + raise ProxyDisabledException("Proxies cannot be used with Advocate") diff --git a/cps/cw_advocate/addrvalidator.py b/cps/cw_advocate/addrvalidator.py new file mode 100644 index 00000000..0f14ce85 --- /dev/null +++ b/cps/cw_advocate/addrvalidator.py @@ -0,0 +1,281 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +import functools +import fnmatch +import ipaddress +import re + +try: + import netifaces + HAVE_NETIFACES = True +except ImportError: + netifaces = None + HAVE_NETIFACES = False + +from .exceptions import NameserverException, ConfigException + + +def canonicalize_hostname(hostname): + """Lowercase and punycodify a hostname""" + # We do the lowercasing after IDNA encoding because we only want to + # lowercase the *ASCII* chars. + # TODO: The differences between IDNA2003 and IDNA2008 might be relevant + # to us, but both specs are damn confusing. + return str(hostname.encode("idna").lower(), 'utf-8') + + +def determine_local_addresses(): + """Get all IPs that refer to this machine according to netifaces""" + if not HAVE_NETIFACES: + raise ConfigException("Tried to determine local addresses, " + "but netifaces module was not importable") + ips = [] + for interface in netifaces.interfaces(): + if_families = netifaces.ifaddresses(interface) + for family_kind in {netifaces.AF_INET, netifaces.AF_INET6}: + addrs = if_families.get(family_kind, []) + for addr in (x.get("addr", "") for x in addrs): + if family_kind == netifaces.AF_INET6: + # We can't do anything sensible with the scope here + addr = addr.split("%")[0] + ips.append(ipaddress.ip_network(addr)) + return ips + + +def add_local_address_arg(func): + """Add the "_local_addresses" kwarg if it's missing + + IMO this information shouldn't be cached between calls (what if one of the + adapters got a new IP at runtime?,) and we don't want each function to + recalculate it. Just recalculate it if the caller didn't provide it for us. + """ + @functools.wraps(func) + def wrapper(self, *args, **kwargs): + if "_local_addresses" not in kwargs: + if self.autodetect_local_addresses: + kwargs["_local_addresses"] = determine_local_addresses() + else: + kwargs["_local_addresses"] = [] + return func(self, *args, **kwargs) + return wrapper + + +class AddrValidator: + _6TO4_RELAY_NET = ipaddress.ip_network("192.88.99.0/24") + # Just the well known prefix, DNS64 servers can set their own + # prefix, but in practice most probably don't. + _DNS64_WK_PREFIX = ipaddress.ip_network("64:ff9b::/96") + DEFAULT_PORT_WHITELIST = {80, 8080, 443, 8443, 8000} + + def __init__( + self, + ip_blacklist=None, + ip_whitelist=None, + port_whitelist=None, + port_blacklist=None, + hostname_blacklist=None, + allow_ipv6=False, + allow_teredo=False, + allow_6to4=False, + allow_dns64=False, + # Must be explicitly set to "False" if you don't want to try + # detecting local interface addresses with netifaces. + autodetect_local_addresses=True, + ): + if not port_blacklist and not port_whitelist: + # An assortment of common HTTPS? ports. + port_whitelist = self.DEFAULT_PORT_WHITELIST.copy() + self.ip_blacklist = ip_blacklist or set() + self.ip_whitelist = ip_whitelist or set() + self.port_blacklist = port_blacklist or set() + self.port_whitelist = port_whitelist or set() + # TODO: ATM this can contain either regexes or globs that are converted + # to regexes upon every check. Create a collection that automagically + # converts them to regexes on insert? + self.hostname_blacklist = hostname_blacklist or set() + self.allow_ipv6 = allow_ipv6 + self.allow_teredo = allow_teredo + self.allow_6to4 = allow_6to4 + self.allow_dns64 = allow_dns64 + self.autodetect_local_addresses = autodetect_local_addresses + + @add_local_address_arg + def is_ip_allowed(self, addr_ip, _local_addresses=None): + if not isinstance(addr_ip, + (ipaddress.IPv4Address, ipaddress.IPv6Address)): + addr_ip = ipaddress.ip_address(addr_ip) + + # The whitelist should take precedence over the blacklist so we can + # punch holes in blacklisted ranges + if any(addr_ip in net for net in self.ip_whitelist): + return True + + if any(addr_ip in net for net in self.ip_blacklist): + return False + + if any(addr_ip in net for net in _local_addresses): + return False + + if addr_ip.version == 4: + if not addr_ip.is_private: + # IPs for carrier-grade NAT. Seems weird that it doesn't set + # `is_private`, but we need to check `not is_global` + if not ipaddress.ip_network(addr_ip).is_global: + return False + elif addr_ip.version == 6: + # You'd better have a good reason for enabling IPv6 + # because Advocate's techniques don't work well without NAT. + if not self.allow_ipv6: + return False + + # v6 addresses can also map to IPv4 addresses! Tricky! + v4_nested = [] + if addr_ip.ipv4_mapped: + v4_nested.append(addr_ip.ipv4_mapped) + # WTF IPv6? Why you gotta have a billion tunneling mechanisms? + # XXX: Do we even really care about these? If we're tunneling + # through public servers we shouldn't be able to access + # addresses on our private network, right? + if addr_ip.sixtofour: + if not self.allow_6to4: + return False + v4_nested.append(addr_ip.sixtofour) + if addr_ip.teredo: + if not self.allow_teredo: + return False + # Check both the client *and* server IPs + v4_nested.extend(addr_ip.teredo) + if addr_ip in self._DNS64_WK_PREFIX: + if not self.allow_dns64: + return False + # When using the well-known prefix the last 4 bytes + # are the IPv4 addr + v4_nested.append(ipaddress.ip_address(addr_ip.packed[-4:])) + + if not all(self.is_ip_allowed(addr_v4) for addr_v4 in v4_nested): + return False + + # fec0::*, apparently deprecated? + if addr_ip.is_site_local: + return False + else: + raise ValueError("Unsupported IP version(?): %r" % addr_ip) + + # 169.254.XXX.XXX, AWS uses these for autoconfiguration + if addr_ip.is_link_local: + return False + # 127.0.0.1, ::1, etc. + if addr_ip.is_loopback: + return False + if addr_ip.is_multicast: + return False + # 192.168.XXX.XXX, 10.XXX.XXX.XXX + if addr_ip.is_private: + return False + # 255.255.255.255, ::ffff:XXXX:XXXX (v6->v4) mapping + if addr_ip.is_reserved: + return False + # There's no reason to connect directly to a 6to4 relay + if addr_ip in self._6TO4_RELAY_NET: + return False + # 0.0.0.0 + if addr_ip.is_unspecified: + return False + + # It doesn't look bad, so... it's must be ok! + return True + + def _hostname_matches_pattern(self, hostname, pattern): + # If they specified a string, just assume they only want basic globbing. + # This stops people from not realizing they're dealing in REs and + # not escaping their periods unless they specifically pass in an RE. + # This has the added benefit of letting us sanely handle globbed + # IDNs by default. + if isinstance(pattern, str): + # convert the glob to a punycode glob, then a regex + pattern = fnmatch.translate(canonicalize_hostname(pattern)) + + hostname = canonicalize_hostname(hostname) + # Down the line the hostname may get treated as a null-terminated string + # (as with `socket.getaddrinfo`.) Try to account for that. + # + # >>> socket.getaddrinfo("example.com\x00aaaa", 80) + # [(2, 1, 6, '', ('93.184.216.34', 80)), [...] + no_null_hostname = hostname.split("\x00")[0] + + return any(re.match(pattern, x.strip(".")) for x + in (no_null_hostname, hostname)) + + def is_hostname_allowed(self, hostname): + # Sometimes (like with "external" services that your IP has privileged + # access to) you might not always know the IP range to blacklist access + # to, or the `A` record might change without you noticing. + # For e.x.: `foocorp.external.org`. + # + # Another option is doing something like: + # + # for addrinfo in socket.getaddrinfo("foocorp.external.org", 80): + # global_validator.ip_blacklist.add(ip_address(addrinfo[4][0])) + # + # but that's not always a good idea if they're behind a third-party lb. + for pattern in self.hostname_blacklist: + if self._hostname_matches_pattern(hostname, pattern): + return False + return True + + @add_local_address_arg + def is_addrinfo_allowed(self, addrinfo, _local_addresses=None): + assert(len(addrinfo) == 5) + # XXX: Do we care about any of the other elements? Guessing not. + family, socktype, proto, canonname, sockaddr = addrinfo + + # The 4th elem inaddrinfo may either be a touple of two or four items, + # depending on whether we're dealing with IPv4 or v6 + if len(sockaddr) == 2: + # v4 + ip, port = sockaddr + elif len(sockaddr) == 4: + # v6 + # XXX: what *are* `flow_info` and `scope_id`? Anything useful? + # Seems like we can figure out all we need about the scope from + # the `is_` properties. + ip, port, flow_info, scope_id = sockaddr + else: + raise ValueError("Unexpected addrinfo format %r" % sockaddr) + + # Probably won't help protect against SSRF, but might prevent our being + # used to attack others' non-HTTP services. See + # http://www.remote.org/jochen/sec/hfpa/ + if self.port_whitelist and port not in self.port_whitelist: + return False + if port in self.port_blacklist: + return False + + if self.hostname_blacklist: + if not canonname: + raise NameserverException( + "addrinfo must contain the canon name to do blacklisting " + "based on hostname. Make sure you use the " + "`socket.AI_CANONNAME` flag, and that each record contains " + "the canon name. Your DNS server might also be garbage." + ) + + if not self.is_hostname_allowed(canonname): + return False + + return self.is_ip_allowed(ip, _local_addresses=_local_addresses) diff --git a/cps/cw_advocate/api.py b/cps/cw_advocate/api.py new file mode 100644 index 00000000..5a5ac203 --- /dev/null +++ b/cps/cw_advocate/api.py @@ -0,0 +1,202 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +""" +advocate.api +~~~~~~~~~~~~ + +This module implements the Requests API, largely a copy/paste from `requests` +itself. + +:copyright: (c) 2015 by Jordan Milne. +:license: Apache2, see LICENSE for more details. + +""" +from collections import OrderedDict +import hashlib +import pickle + +from requests import Session as RequestsSession + +# import cw_advocate +from .adapters import ValidatingHTTPAdapter +from .exceptions import MountDisabledException + + +class Session(RequestsSession): + """Convenience wrapper around `requests.Session` set up for `advocate`ing""" + + __attrs__ = RequestsSession.__attrs__ + ["validator"] + DEFAULT_VALIDATOR = None + """ + User-replaceable default validator to use for all Advocate sessions, + includes sessions created by advocate.get() + """ + + def __init__(self, *args, **kwargs): + self.validator = kwargs.pop("validator", None) or self.DEFAULT_VALIDATOR + adapter_kwargs = kwargs.pop("_adapter_kwargs", {}) + + # `Session.__init__()` calls `mount()` internally, so we need to allow + # it temporarily + self.__mount_allowed = True + RequestsSession.__init__(self, *args, **kwargs) + + # Drop any existing adapters + self.adapters = OrderedDict() + + self.mount("http://", ValidatingHTTPAdapter(validator=self.validator, **adapter_kwargs)) + self.mount("https://", ValidatingHTTPAdapter(validator=self.validator, **adapter_kwargs)) + self.__mount_allowed = False + + def mount(self, *args, **kwargs): + """Wrapper around `mount()` to prevent a protection bypass""" + if self.__mount_allowed: + super().mount(*args, **kwargs) + else: + raise MountDisabledException( + "mount() is disabled to prevent protection bypasses" + ) + + +def session(*args, **kwargs): + return Session(*args, **kwargs) + + +def request(method, url, **kwargs): + """Constructs and sends a :class:`Request `. + + :param method: method for the new :class:`Request` object. + :param url: URL for the new :class:`Request` object. + :param params: (optional) Dictionary or bytes to be sent in the query string for the :class:`Request`. + :param data: (optional) Dictionary, bytes, or file-like object to send in the body of the :class:`Request`. + :param json: (optional) json data to send in the body of the :class:`Request`. + :param headers: (optional) Dictionary of HTTP Headers to send with the :class:`Request`. + :param cookies: (optional) Dict or CookieJar object to send with the :class:`Request`. + :param files: (optional) Dictionary of ``'name': file-like-objects`` (or ``{'name': ('filename', fileobj)}``) for multipart encoding upload. + :param auth: (optional) Auth tuple to enable Basic/Digest/Custom HTTP Auth. + :param timeout: (optional) How long to wait for the server to send data + before giving up, as a float, or a (`connect timeout, read timeout + `_) tuple. + :type timeout: float or tuple + :param allow_redirects: (optional) Boolean. Set to True if POST/PUT/DELETE redirect following is allowed. + :type allow_redirects: bool + :param proxies: (optional) Dictionary mapping protocol to the URL of the proxy. + :param verify: (optional) if ``True``, the SSL cert will be verified. A CA_BUNDLE path can also be provided. + :param stream: (optional) if ``False``, the response content will be immediately downloaded. + :param cert: (optional) if String, path to ssl client cert file (.pem). If Tuple, ('cert', 'key') pair. + :return: :class:`Response ` object + :rtype: requests.Response + """ + + validator = kwargs.pop("validator", None) + with Session(validator=validator) as sess: + response = sess.request(method=method, url=url, **kwargs) + return response + + +def get(url, **kwargs): + """Sends a GET request. + + :param url: URL for the new :class:`Request` object. + :param **kwargs: Optional arguments that ``request`` takes. + :return: :class:`Response ` object + :rtype: requests.Response + """ + + kwargs.setdefault('allow_redirects', True) + return request('get', url, **kwargs) + + +class RequestsAPIWrapper: + """Provides a `requests.api`-like interface with a specific validator""" + + # Due to how the classes are dynamically constructed pickling may not work + # correctly unless loaded within the same interpreter instance. + # Enable at your peril. + SUPPORT_WRAPPER_PICKLING = False + + def __init__(self, validator): + # Do this here to avoid circular import issues + try: + from .futures import FuturesSession + have_requests_futures = True + except ImportError as e: + have_requests_futures = False + + self.validator = validator + outer_self = self + + class _WrappedSession(Session): + """An `advocate.Session` that uses the wrapper's blacklist + + the wrapper is meant to be a transparent replacement for `requests`, + so people should be able to subclass `wrapper.Session` and still + get the desired validation behaviour + """ + DEFAULT_VALIDATOR = outer_self.validator + + self._make_wrapper_cls_global(_WrappedSession) + + if have_requests_futures: + + class _WrappedFuturesSession(FuturesSession): + """Like _WrappedSession, but for `FuturesSession`s""" + DEFAULT_VALIDATOR = outer_self.validator + self._make_wrapper_cls_global(_WrappedFuturesSession) + + self.FuturesSession = _WrappedFuturesSession + + self.request = self._default_arg_wrapper(request) + self.get = self._default_arg_wrapper(get) + self.Session = _WrappedSession + + def __getattr__(self, item): + # This class is meant to mimic the requests base module, so if we don't + # have this attribute, it might be on the base module (like the Request + # class, etc.) + try: + return object.__getattribute__(self, item) + except AttributeError: + from . import cw_advocate + return getattr(cw_advocate, item) + + def _default_arg_wrapper(self, fun): + def wrapped_func(*args, **kwargs): + kwargs.setdefault("validator", self.validator) + return fun(*args, **kwargs) + return wrapped_func + + def _make_wrapper_cls_global(self, cls): + if not self.SUPPORT_WRAPPER_PICKLING: + return + # Gnarly, but necessary to give pickle a consistent module-level + # reference for each wrapper. + wrapper_hash = hashlib.sha256(pickle.dumps(self)).hexdigest() + cls.__name__ = "_".join((cls.__name__, wrapper_hash)) + cls.__qualname__ = ".".join((__name__, cls.__name__)) + if not globals().get(cls.__name__): + globals()[cls.__name__] = cls + + +__all__ = ( + "get", + "request", + "session", + "Session", + "RequestsAPIWrapper", +) diff --git a/cps/cw_advocate/connection.py b/cps/cw_advocate/connection.py new file mode 100644 index 00000000..ce790ada --- /dev/null +++ b/cps/cw_advocate/connection.py @@ -0,0 +1,201 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +import ipaddress +import socket +from socket import timeout as SocketTimeout + +from urllib3.connection import HTTPSConnection, HTTPConnection +from urllib3.exceptions import ConnectTimeoutError +from urllib3.util.connection import _set_socket_options +from urllib3.util.connection import create_connection as old_create_connection + +from . import addrvalidator +from .exceptions import UnacceptableAddressException + + +def advocate_getaddrinfo(host, port, get_canonname=False): + addrinfo = socket.getaddrinfo( + host, + port, + 0, + socket.SOCK_STREAM, + 0, + # We need what the DNS client sees the hostname as, correctly handles + # IDNs and tricky things like `private.foocorp.org\x00.google.com`. + # All IDNs will be converted to punycode. + socket.AI_CANONNAME if get_canonname else 0, + ) + return fix_addrinfo(addrinfo) + + +def fix_addrinfo(records): + """ + Propagate the canonname across records and parse IPs + + I'm not sure if this is just the behaviour of `getaddrinfo` on Linux, but + it seems like only the first record in the set has the canonname field + populated. + """ + def fix_record(record, canonname): + sa = record[4] + sa = (ipaddress.ip_address(sa[0]),) + sa[1:] + return record[0], record[1], record[2], canonname, sa + + canonname = None + if records: + # Apparently the canonical name is only included in the first record? + # Add it to all of them. + assert(len(records[0]) == 5) + canonname = records[0][3] + return tuple(fix_record(x, canonname) for x in records) + + +# Lifted from requests' urllib3, which in turn lifted it from `socket.py`. Oy! +def validating_create_connection(address, + timeout=socket._GLOBAL_DEFAULT_TIMEOUT, + source_address=None, socket_options=None, + validator=None): + """Connect to *address* and return the socket object. + + Convenience function. Connect to *address* (a 2-tuple ``(host, + port)``) and return the socket object. Passing the optional + *timeout* parameter will set the timeout on the socket instance + before attempting to connect. If no *timeout* is supplied, the + global default timeout setting returned by :func:`getdefaulttimeout` + is used. If *source_address* is set it must be a tuple of (host, port) + for the socket to bind as a source address before making the connection. + An host of '' or port 0 tells the OS to use the default. + """ + + host, port = address + # We can skip asking for the canon name if we're not doing hostname-based + # blacklisting. + need_canonname = False + if validator.hostname_blacklist: + need_canonname = True + # We check both the non-canonical and canonical hostnames so we can + # catch both of these: + # CNAME from nonblacklisted.com -> blacklisted.com + # CNAME from blacklisted.com -> nonblacklisted.com + if not validator.is_hostname_allowed(host): + raise UnacceptableAddressException(host) + + err = None + addrinfo = advocate_getaddrinfo(host, port, get_canonname=need_canonname) + if addrinfo: + if validator.autodetect_local_addresses: + local_addresses = addrvalidator.determine_local_addresses() + else: + local_addresses = [] + for res in addrinfo: + # Are we allowed to connect with this result? + if not validator.is_addrinfo_allowed( + res, + _local_addresses=local_addresses, + ): + continue + af, socktype, proto, canonname, sa = res + # Unparse the validated IP + sa = (sa[0].exploded,) + sa[1:] + sock = None + try: + sock = socket.socket(af, socktype, proto) + + # If provided, set socket level options before connecting. + # This is the only addition urllib3 makes to this function. + _set_socket_options(sock, socket_options) + + if timeout is not socket._GLOBAL_DEFAULT_TIMEOUT: + sock.settimeout(timeout) + if source_address: + sock.bind(source_address) + sock.connect(sa) + return sock + + except socket.error as _: + err = _ + if sock is not None: + sock.close() + sock = None + + if err is None: + # If we got here, none of the results were acceptable + err = UnacceptableAddressException(address) + if err is not None: + raise err + else: + raise socket.error("getaddrinfo returns an empty list") + + +# TODO: Is there a better way to add this to multiple classes with different +# base classes? I tried a mixin, but it used the base method instead. +def _validating_new_conn(self): + """ Establish a socket connection and set nodelay settings on it. + + :return: New socket connection. + """ + extra_kw = {} + if self.source_address: + extra_kw['source_address'] = self.source_address + + if self.socket_options: + extra_kw['socket_options'] = self.socket_options + + try: + # Hack around HTTPretty's patched sockets + # TODO: some better method of hacking around it that checks if we + # _would have_ connected to a private addr? + conn_func = validating_create_connection + if socket.getaddrinfo.__module__.startswith("httpretty"): + conn_func = old_create_connection + else: + extra_kw["validator"] = self._validator + + conn = conn_func( + (self.host, self.port), + self.timeout, + **extra_kw + ) + + except SocketTimeout: + raise ConnectTimeoutError( + self, "Connection to %s timed out. (connect timeout=%s)" % + (self.host, self.timeout)) + + return conn + + +# Don't silently break if the private API changes across urllib3 versions +assert(hasattr(HTTPConnection, '_new_conn')) +assert(hasattr(HTTPSConnection, '_new_conn')) + + +class ValidatingHTTPConnection(HTTPConnection): + _new_conn = _validating_new_conn + + def __init__(self, *args, **kwargs): + self._validator = kwargs.pop("validator") + HTTPConnection.__init__(self, *args, **kwargs) + + +class ValidatingHTTPSConnection(HTTPSConnection): + _new_conn = _validating_new_conn + + def __init__(self, *args, **kwargs): + self._validator = kwargs.pop("validator") + HTTPSConnection.__init__(self, *args, **kwargs) diff --git a/cps/cw_advocate/connectionpool.py b/cps/cw_advocate/connectionpool.py new file mode 100644 index 00000000..3bbbfac7 --- /dev/null +++ b/cps/cw_advocate/connectionpool.py @@ -0,0 +1,39 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +from urllib3 import HTTPConnectionPool, HTTPSConnectionPool + +from .connection import ( + ValidatingHTTPConnection, + ValidatingHTTPSConnection, +) + +# Don't silently break if the private API changes across urllib3 versions +assert(hasattr(HTTPConnectionPool, 'ConnectionCls')) +assert(hasattr(HTTPSConnectionPool, 'ConnectionCls')) +assert(hasattr(HTTPConnectionPool, 'scheme')) +assert(hasattr(HTTPSConnectionPool, 'scheme')) + + +class ValidatingHTTPConnectionPool(HTTPConnectionPool): + scheme = 'http' + ConnectionCls = ValidatingHTTPConnection + + +class ValidatingHTTPSConnectionPool(HTTPSConnectionPool): + scheme = 'https' + ConnectionCls = ValidatingHTTPSConnection diff --git a/cps/cw_advocate/exceptions.py b/cps/cw_advocate/exceptions.py new file mode 100644 index 00000000..5ff9852b --- /dev/null +++ b/cps/cw_advocate/exceptions.py @@ -0,0 +1,39 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +class AdvocateException(Exception): + pass + + +class UnacceptableAddressException(AdvocateException): + pass + + +class NameserverException(AdvocateException): + pass + + +class MountDisabledException(AdvocateException): + pass + + +class ProxyDisabledException(NotImplementedError, AdvocateException): + pass + + +class ConfigException(AdvocateException): + pass diff --git a/cps/cw_advocate/poolmanager.py b/cps/cw_advocate/poolmanager.py new file mode 100644 index 00000000..d912d65d --- /dev/null +++ b/cps/cw_advocate/poolmanager.py @@ -0,0 +1,61 @@ +# +# Copyright 2015 Jordan Milne +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# Source: https://github.com/JordanMilne/Advocate + +import collections +import functools + +from urllib3 import PoolManager +from urllib3.poolmanager import _default_key_normalizer, PoolKey + +from .connectionpool import ( + ValidatingHTTPSConnectionPool, + ValidatingHTTPConnectionPool, +) + +pool_classes_by_scheme = { + "http": ValidatingHTTPConnectionPool, + "https": ValidatingHTTPSConnectionPool, +} + +AdvocatePoolKey = collections.namedtuple('AdvocatePoolKey', + PoolKey._fields + ('key_validator',)) + + +def key_normalizer(key_class, request_context): + request_context = request_context.copy() + # TODO: add ability to serialize validator rules to dict, + # allowing pool to be shared between sessions with the same + # rules. + request_context["validator"] = id(request_context["validator"]) + return _default_key_normalizer(key_class, request_context) + + +key_fn_by_scheme = { + 'http': functools.partial(key_normalizer, AdvocatePoolKey), + 'https': functools.partial(key_normalizer, AdvocatePoolKey), +} + + +class ValidatingPoolManager(PoolManager): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + + # Make sure the API hasn't changed + assert (hasattr(self, 'pool_classes_by_scheme')) + + self.pool_classes_by_scheme = pool_classes_by_scheme + self.key_fn_by_scheme = key_fn_by_scheme.copy() diff --git a/cps/babel.py b/cps/cw_babel.py similarity index 93% rename from cps/babel.py rename to cps/cw_babel.py index cf043d52..878984f6 100644 --- a/cps/babel.py +++ b/cps/cw_babel.py @@ -34,7 +34,7 @@ def get_user_locale_language(user_language): def get_available_locale(): - return [Locale('en')] + babel.list_translations() + return sorted(babel.list_translations(), key=lambda x: x.display_name.lower()) def get_available_translations(): diff --git a/cps/cw_login/login_manager.py b/cps/cw_login/login_manager.py index 21dc803d..a3714af6 100644 --- a/cps/cw_login/login_manager.py +++ b/cps/cw_login/login_manager.py @@ -1,4 +1,5 @@ from datetime import datetime +from datetime import timezone from datetime import timedelta import hashlib @@ -496,7 +497,7 @@ class LoginManager: duration = timedelta(seconds=duration) try: - expires = datetime.utcnow() + duration + expires = datetime.now(timezone.utc) + duration except TypeError as e: raise Exception( "REMEMBER_COOKIE_DURATION must be a datetime.timedelta," diff --git a/cps/db.py b/cps/db.py index 62b68b47..15921d7a 100644 --- a/cps/db.py +++ b/cps/db.py @@ -20,10 +20,11 @@ import os import re import json -from datetime import datetime +from datetime import datetime, timezone from urllib.parse import quote import unidecode -from weakref import WeakSet +# from weakref import WeakSet +from uuid import uuid4 from sqlite3 import OperationalError as sqliteOperationalError from sqlalchemy import create_engine @@ -44,11 +45,11 @@ from sqlalchemy.ext.associationproxy import association_proxy from .cw_login import current_user from flask_babel import gettext as _ from flask_babel import get_locale -from flask import flash +from flask import flash, g, Flask from . import logger, ub, isoLanguages from .pagination import Pagination - +from .string_helper import strip_whitespaces log = logger.create() @@ -101,6 +102,16 @@ class Identifiers(Base): type = Column(String(collation='NOCASE'), nullable=False, default="isbn") val = Column(String(collation='NOCASE'), nullable=False) book = Column(Integer, ForeignKey('books.id'), nullable=False) + amazon = { + "jp": "co.jp", + "uk": "co.uk", + "us": "com", + "au": "com.au", + "be": "com.be", + "br": "com.br", + "tr": "com.tr", + "mx": "com.mx", + } def __init__(self, val, id_type, book): super().__init__() @@ -113,7 +124,11 @@ class Identifiers(Base): if format_type == 'amazon': return "Amazon" elif format_type.startswith("amazon_"): - return "Amazon.{0}".format(format_type[7:]) + label_amazon = "Amazon.{0}" + country_code = format_type[7:].lower() + if country_code not in self.amazon: + return label_amazon.format(country_code) + return label_amazon.format(self.amazon[country_code]) elif format_type == "isbn": return "ISBN" elif format_type == "doi": @@ -136,6 +151,8 @@ class Identifiers(Base): return "ISSN" elif format_type == "isfdb": return "ISFDB" + elif format_type == "storygraph": + return "StoryGraph" if format_type == "lubimyczytac": return "Lubimyczytac" if format_type == "databazeknih": @@ -148,7 +165,11 @@ class Identifiers(Base): if format_type == "amazon" or format_type == "asin": return "https://amazon.com/dp/{0}".format(self.val) elif format_type.startswith('amazon_'): - return "https://amazon.{0}/dp/{1}".format(format_type[7:], self.val) + link_amazon = "https://amazon.{0}/dp/{1}" + country_code = format_type[7:].lower() + if country_code not in self.amazon: + return link_amazon.format(country_code, self.val) + return link_amazon.format(self.amazon[country_code], self.val) elif format_type == "isbn": return "https://www.worldcat.org/isbn/{0}".format(self.val) elif format_type == "doi": @@ -172,9 +193,11 @@ class Identifiers(Base): elif format_type == "issn": return "https://portal.issn.org/resource/ISSN/{0}".format(self.val) elif format_type == "isfdb": - return "http://www.isfdb.org/cgi-bin/pl.cgi?{0}".format(self.val) + return "https://www.isfdb.org/cgi-bin/pl.cgi?{0}".format(self.val) elif format_type == "databazeknih": return "https://www.databazeknih.cz/knihy/{0}".format(self.val) + elif format_type == "storygraph": + return "https://app.thestorygraph.com/books/{0}".format(self.val) elif self.val.lower().startswith("javascript:"): return quote(self.val) elif self.val.lower().startswith("data:"): @@ -378,10 +401,10 @@ class Books(Base): title = Column(String(collation='NOCASE'), nullable=False, default='Unknown') sort = Column(String(collation='NOCASE')) author_sort = Column(String(collation='NOCASE')) - timestamp = Column(TIMESTAMP, default=datetime.utcnow) + timestamp = Column(TIMESTAMP, default=lambda: datetime.now(timezone.utc)) pubdate = Column(TIMESTAMP, default=DEFAULT_PUBDATE) series_index = Column(String, nullable=False, default="1.0") - last_modified = Column(TIMESTAMP, default=datetime.utcnow) + last_modified = Column(TIMESTAMP, default=lambda: datetime.now(timezone.utc)) path = Column(String, default="", nullable=False) has_cover = Column(Integer, default=0) uuid = Column(String) @@ -509,34 +532,25 @@ class AlchemyEncoder(json.JSONEncoder): class CalibreDB: - _init = False - engine = None config = None - session_factory = None - # This is a WeakSet so that references here don't keep other CalibreDB - # instances alive once they reach the end of their respective scopes - instances = WeakSet() + config_calibre_dir = None + app_db_path = None - def __init__(self, expire_on_commit=True, init=False): + def __init__(self, _app: Flask=None): # , expire_on_commit=True, init=False): """ Initialize a new CalibreDB session """ - self.session = None - if init: - self.init_db(expire_on_commit) + self.Session = None + #if init: + # self.init_db(expire_on_commit) + if _app is not None and not _app._got_first_request: + self.init_app(_app) - def init_db(self, expire_on_commit=True): - if self._init: - self.init_session(expire_on_commit) - - self.instances.add(self) - - def init_session(self, expire_on_commit=True): - self.session = self.session_factory() - self.session.expire_on_commit = expire_on_commit - self.update_title_sort(self.config) + def init_app(self, _app): + _app.teardown_appcontext(self.teardown) @classmethod def setup_db_cc_classes(cls, cc): + global cc_classes cc_ids = [] books_custom_column_links = {} for row in cc: @@ -604,8 +618,6 @@ class CalibreDB: secondary=books_custom_column_links[cc_id[0]], backref='books')) - return cc_classes - @classmethod def check_valid_db(cls, config_calibre_dir, app_db_path, config_calibre_uuid): if not config_calibre_dir: @@ -625,7 +637,6 @@ class CalibreDB: local_session = scoped_session(sessionmaker()) local_session.configure(bind=connection) database_uuid = local_session().query(Library_Id).one_or_none() - # local_session.dispose() check_engine.connect() db_change = config_calibre_uuid != database_uuid.uuid @@ -633,13 +644,30 @@ class CalibreDB: return False, False return True, db_change + def teardown(self, exception): + ctx = g.get("lib_sql") + if ctx: + ctx.close() + + @property + def session(self): + # connect or get active connection + if not g.get("lib_sql"): + g.lib_sql = self.connect() + return g.lib_sql + @classmethod - def update_config(cls, config): + def update_config(cls, config, config_calibre_dir, app_db_path): cls.config = config + cls.config_calibre_dir = config_calibre_dir + cls.app_db_path = app_db_path + + + def connect(self): + return self.setup_db(self.config_calibre_dir, self.app_db_path) @classmethod def setup_db(cls, config_calibre_dir, app_db_path): - cls.dispose() if not config_calibre_dir: cls.config.invalidate() @@ -651,16 +679,17 @@ class CalibreDB: return None try: - cls.engine = create_engine('sqlite://', + engine = create_engine('sqlite://', echo=False, isolation_level="SERIALIZABLE", connect_args={'check_same_thread': False}, poolclass=StaticPool) - with cls.engine.begin() as connection: + with engine.begin() as connection: + connection.execute(text('PRAGMA cache_size = 10000;')) connection.execute(text("attach database '{}' as calibre;".format(dbpath))) connection.execute(text("attach database '{}' as app_settings;".format(app_db_path))) - conn = cls.engine.connect() + conn = engine.connect() # conn.text_factory = lambda b: b.decode(errors = 'ignore') possible fix for #1302 except Exception as ex: cls.config.invalidate(ex) @@ -676,13 +705,10 @@ class CalibreDB: log.error_or_exception(e) return None - cls.session_factory = scoped_session(sessionmaker(autocommit=False, - autoflush=True, - bind=cls.engine, future=True)) - for inst in cls.instances: - inst.init_session() + return scoped_session(sessionmaker(autocommit=False, + autoflush=False, + bind=engine, future=True)) - cls._init = True def get_book(self, book_id): return self.session.query(Books).filter(Books.id == book_id).first() @@ -875,10 +901,12 @@ class CalibreDB: authors_ordered = list() # error = False for auth in sort_authors: - results = self.session.query(Authors).filter(Authors.sort == auth.lstrip().strip()).all() + auth = strip_whitespaces(auth) + results = self.session.query(Authors).filter(Authors.sort == auth).all() # ToDo: How to handle not found author name if not len(results): - log.error("Author {} not found to display name in right order".format(auth.strip())) + book_id = entry.id if isinstance(entry, Books) else entry[0].id + log.error("Author '{}' of book {} not found to display name in right order".format(auth, book_id)) # error = True break for r in results: @@ -900,7 +928,8 @@ class CalibreDB: def get_typeahead(self, database, query, replace=('', ''), tag_filter=true()): query = query or '' - self.session.connection().connection.connection.create_function("lower", 1, lcase) + self.create_functions() + # self.session.connection().connection.connection.create_function("lower", 1, lcase) entries = self.session.query(database).filter(tag_filter). \ filter(func.lower(database.name).ilike("%" + query + "%")).all() # json_dumps = json.dumps([dict(name=escape(r.name.replace(*replace))) for r in entries]) @@ -908,7 +937,8 @@ class CalibreDB: return json_dumps def check_exists_book(self, authr, title): - self.session.connection().connection.connection.create_function("lower", 1, lcase) + self.create_functions() + # self.session.connection().connection.connection.create_function("lower", 1, lcase) q = list() author_terms = re.split(r'\s*&\s*', authr) for author_term in author_terms: @@ -918,8 +948,9 @@ class CalibreDB: .filter(and_(Books.authors.any(and_(*q)), func.lower(Books.title).ilike("%" + title + "%"))).first() def search_query(self, term, config, *join): - term.strip().lower() - self.session.connection().connection.connection.create_function("lower", 1, lcase) + strip_whitespaces(term).lower() + self.create_functions() + # self.session.connection().connection.connection.create_function("lower", 1, lcase) q = list() author_terms = re.split("[, ]+", term) for author_term in author_terms: @@ -1017,7 +1048,7 @@ class CalibreDB: lang.name = isoLanguages.get_language_name(get_locale(), lang.lang_code) return sorted(languages, key=lambda x: x.name, reverse=reverse_order) - def update_title_sort(self, config, conn=None): + def create_functions(self, config=None): # user defined sort function for calibre databases (Series, etc.) def _title_sort(title): # calibre sort stuff @@ -1026,56 +1057,27 @@ class CalibreDB: if match: prep = match.group(1) title = title[len(prep):] + ', ' + prep - return title.strip() + return strip_whitespaces(title) try: - # sqlalchemy <1.4.24 - conn = conn or self.session.connection().connection.driver_connection + # sqlalchemy <1.4.24 and sqlalchemy 2.0 + conn = self.session.connection().connection.driver_connection except AttributeError: - # sqlalchemy >1.4.24 and sqlalchemy 2.0 - conn = conn or self.session.connection().connection.connection + # sqlalchemy >1.4.24 + conn = self.session.connection().connection.connection try: - conn.create_function("title_sort", 1, _title_sort) + if config: + conn.create_function("title_sort", 1, _title_sort) + conn.create_function('uuid4', 0, lambda: str(uuid4())) + conn.create_function("lower", 1, lcase) except sqliteOperationalError: pass - @classmethod - def dispose(cls): - # global session - - for inst in cls.instances: - old_session = inst.session - inst.session = None - if old_session: - try: - old_session.close() - except Exception: - pass - if old_session.bind: - try: - old_session.bind.dispose() - except Exception: - pass - - for attr in list(Books.__dict__.keys()): - if attr.startswith("custom_column_"): - setattr(Books, attr, None) - - for db_class in cc_classes.values(): - Base.metadata.remove(db_class.__table__) - cc_classes.clear() - - for table in reversed(Base.metadata.sorted_tables): - name = table.key - if name.startswith("custom_column_") or name.startswith("books_custom_column_"): - if table is not None: - Base.metadata.remove(table) - def reconnect_db(self, config, app_db_path): - self.dispose() - self.engine.dispose() + # self.dispose() + # self.engine.dispose() self.setup_db(config.config_calibre_dir, app_db_path) - self.update_config(config) + self.update_config(config, config.config_calibre_dir, app_db_path) def lcase(s): diff --git a/cps/debug_info.py b/cps/debug_info.py index 879846ab..66762e1c 100644 --- a/cps/debug_info.py +++ b/cps/debug_info.py @@ -23,10 +23,10 @@ import zipfile import json from io import BytesIO from flask_babel.speaklater import LazyString - +from importlib.metadata import metadata import os -from flask import send_file, __version__ +from flask import send_file from . import logger, config from .about import collect_stats @@ -49,7 +49,8 @@ def assemble_logfiles(file_name): with open(f, 'rb') as fd: shutil.copyfileobj(fd, wfd) wfd.seek(0) - if int(__version__.split('.')[0]) < 2: + version = metadata("flask")["Version"] + if int(version.split('.')[0]) < 2: return send_file(wfd, as_attachment=True, attachment_filename=os.path.basename(file_name)) @@ -72,7 +73,8 @@ def send_debug(): for fp in file_list: zf.write(fp, os.path.basename(fp)) memory_zip.seek(0) - if int(__version__.split('.')[0]) < 2: + version = metadata("flask")["Version"] + if int(version.split('.')[0]) < 2: return send_file(memory_zip, as_attachment=True, attachment_filename="Calibre-Web-debug-pack.zip") diff --git a/cps/dep_check.py b/cps/dep_check.py index b1917f7a..f1c290ad 100644 --- a/cps/dep_check.py +++ b/cps/dep_check.py @@ -39,8 +39,24 @@ def load_dependencies(optional=False): with open(req_path, 'r') as f: for line in f: if not line.startswith('#') and not line == '\n' and not line.startswith('git'): - res = re.match(r'(.*?)([<=>\s]+)([\d\.]+),?\s?([<=>\s]+)?([\d\.]+)?', line.strip()) + res = re.match(r'(.*?)([<=>\s]+)([\d\.]+),?\s?([<=>\s]+)?([\d\.]+)?(?:\s?;\s?' + r'(?:(python_version)\s?([<=>]+)\s?\'([\d\.]+)\'|' + r'(sys_platform)\s?([\!=]+)\s?\'([\w]+)\'))?', line.strip()) try: + if res.group(7) and res.group(8): + val = res.group(8).split(".") + if not eval(str(sys.version_info[0]) + "." + "{:02d}".format(sys.version_info[1]) + + res.group(7) + val[0] + "." + "{:02d}".format(int(val[1]))): + continue + elif res.group(10) and res.group(11): + # only installed if platform is eqal, don't check if platform is not equal + if res.group(10) == "==": + if sys.platform != res.group(11): + continue + # installed if platform is not eqal, don't check if platform is equal + elif res.group(10) == "!=": + if sys.platform == res.group(11): + continue if getattr(sys, 'frozen', False): dep_version = exe_deps[res.group(1).lower().replace('_', '-')] else: @@ -63,7 +79,7 @@ def dependency_check(optional=False): deps = load_dependencies(optional) for dep in deps: try: - dep_version_int = [int(x) if x.isnumeric() else 0 for x in dep[0].split('.')] + dep_version_int = [int(x) if x.isnumeric() else 0 for x in dep[0].split('.')[:3]] low_check = [int(x) for x in dep[3].split('.')] high_check = [int(x) for x in dep[5].split('.')] except AttributeError: diff --git a/cps/editbooks.py b/cps/editbooks.py index 5f58cf81..948ad1ab 100644 --- a/cps/editbooks.py +++ b/cps/editbooks.py @@ -21,18 +21,18 @@ # along with this program. If not, see . import os -from datetime import datetime +from datetime import datetime, timezone import json from shutil import copyfile -from uuid import uuid4 + from markupsafe import escape, Markup # dependency of flask from functools import wraps -from flask import Blueprint, request, flash, redirect, url_for, abort, Response +from flask import Blueprint, request, flash, redirect, url_for, abort, jsonify, make_response, Response from flask_babel import gettext as _ from flask_babel import lazy_gettext as N_ from flask_babel import get_locale -from .cw_login import current_user, login_required +from .cw_login import current_user from sqlalchemy.exc import OperationalError, IntegrityError, InterfaceError from sqlalchemy.orm.exc import StaleDataError from sqlalchemy.sql.expression import func @@ -47,7 +47,7 @@ from .kobo_sync_status import change_archived_books from .redirect import get_redirect_location from .file_helper import validate_mime_type from .usermanagement import user_login_required, login_required_if_no_ano - +from .string_helper import strip_whitespaces editbook = Blueprint('edit-book', __name__) log = logger.create() @@ -76,7 +76,7 @@ def edit_required(f): @editbook.route("/ajax/delete/", methods=["POST"]) @user_login_required def delete_book_from_details(book_id): - return Response(delete_book_from_table(book_id, "", True), mimetype='application/json') + return delete_book_from_table(book_id, "", True) @editbook.route("/delete/", defaults={'book_format': ""}, methods=["POST"]) @@ -97,157 +97,22 @@ def show_edit_book(book_id): @login_required_if_no_ano @edit_required def edit_book(book_id): - modify_date = False - edit_error = False - - # create the function for sorting... - calibre_db.update_title_sort(config) - - book = calibre_db.get_filtered_book(book_id, allow_show_archived=True) - # Book not found - if not book: - flash(_("Oops! Selected book is unavailable. File does not exist or is not accessible"), - category="error") - return redirect(url_for("web.index")) - - to_save = request.form.to_dict() - - try: - # Update folder of book on local disk - edited_books_id = None - title_author_error = None - # handle book title change - title_change = handle_title_on_edit(book, to_save["book_title"]) - # handle book author change - input_authors, author_change = handle_author_on_edit(book, to_save["author_name"]) - if author_change or title_change: - edited_books_id = book.id - modify_date = True - title_author_error = helper.update_dir_structure(edited_books_id, - config.get_book_path(), - input_authors[0]) - if title_author_error: - flash(title_author_error, category="error") - calibre_db.session.rollback() - book = calibre_db.get_filtered_book(book_id, allow_show_archived=True) - - # handle upload other formats from local disk - meta = upload_single_file(request, book, book_id) - # only merge metadata if file was uploaded and no error occurred (meta equals not false or none) - upload_format = False - if meta: - upload_format = merge_metadata(to_save, meta) - # handle upload covers from local disk - cover_upload_success = upload_cover(request, book) - if cover_upload_success: - book.has_cover = 1 - modify_date = True - - # upload new covers or new file formats to google drive - if config.config_use_google_drive: - gdriveutils.updateGdriveCalibreFromLocal() - - if to_save.get("cover_url", None): - if not current_user.role_upload(): - edit_error = True - flash(_("User has no rights to upload cover"), category="error") - if to_save["cover_url"].endswith('/static/generic_cover.jpg'): - book.has_cover = 0 - else: - result, error = helper.save_cover_from_url(to_save["cover_url"].strip(), book.path) - if result is True: - book.has_cover = 1 - modify_date = True - helper.replace_cover_thumbnail_cache(book.id) - else: - flash(error, category="error") - - # Add default series_index to book - modify_date |= edit_book_series_index(to_save["series_index"], book) - # Handle book comments/description - modify_date |= edit_book_comments(Markup(to_save['description']).unescape(), book) - # Handle identifiers - input_identifiers = identifier_list(to_save, book) - modification, warning = modify_identifiers(input_identifiers, book.identifiers, calibre_db.session) - if warning: - flash(_("Identifiers are not Case Sensitive, Overwriting Old Identifier"), category="warning") - modify_date |= modification - # Handle book tags - modify_date |= edit_book_tags(to_save['tags'], book) - # Handle book series - modify_date |= edit_book_series(to_save["series"], book) - # handle book publisher - modify_date |= edit_book_publisher(to_save['publisher'], book) - # handle book languages - try: - modify_date |= edit_book_languages(to_save['languages'], book, upload_format) - except ValueError as e: - flash(str(e), category="error") - edit_error = True - # handle book ratings - modify_date |= edit_book_ratings(to_save, book) - # handle cc data - modify_date |= edit_all_cc_data(book_id, book, to_save) - - if to_save.get("pubdate", None): - try: - book.pubdate = datetime.strptime(to_save["pubdate"], "%Y-%m-%d") - except ValueError as e: - book.pubdate = db.Books.DEFAULT_PUBDATE - flash(str(e), category="error") - edit_error = True - else: - book.pubdate = db.Books.DEFAULT_PUBDATE - - if modify_date: - book.last_modified = datetime.utcnow() - kobo_sync_status.remove_synced_book(edited_books_id, all=True) - calibre_db.set_metadata_dirty(book.id) - - calibre_db.session.merge(book) - calibre_db.session.commit() - if config.config_use_google_drive: - gdriveutils.updateGdriveCalibreFromLocal() - if meta is not False \ - and edit_error is not True \ - and title_author_error is not True \ - and cover_upload_success is not False: - flash(_("Metadata successfully updated"), category="success") - if "detail_view" in to_save: - return redirect(url_for('web.show_book', book_id=book.id)) - else: - return render_edit_book(book_id) - except ValueError as e: - log.error_or_exception("Error: {}".format(e)) - calibre_db.session.rollback() - flash(str(e), category="error") - return redirect(url_for('web.show_book', book_id=book.id)) - except (OperationalError, IntegrityError, StaleDataError, InterfaceError) as e: - log.error_or_exception("Database error: {}".format(e)) - calibre_db.session.rollback() - flash(_("Oops! Database Error: %(error)s.", error=e.orig if hasattr(e, "orig") else e), category="error") - return redirect(url_for('web.show_book', book_id=book.id)) - except Exception as ex: - log.error_or_exception(ex) - calibre_db.session.rollback() - flash(_("Error editing book: {}".format(ex)), category="error") - return redirect(url_for('web.show_book', book_id=book.id)) + return do_edit_book(book_id) @editbook.route("/upload", methods=["POST"]) @login_required_if_no_ano @upload_required def upload(): - if not config.config_uploading: - abort(404) - if request.method == 'POST' and 'btn-upload' in request.files: + if len(request.files.getlist("btn-upload-format")): + book_id = request.form.get('book_id', -1) + return do_edit_book(book_id, request.files.getlist("btn-upload-format")) + elif len(request.files.getlist("btn-upload")): for requested_file in request.files.getlist("btn-upload"): try: modify_date = False # create the function for sorting... - calibre_db.update_title_sort(config) - calibre_db.session.connection().connection.connection.create_function('uuid4', 0, lambda: str(uuid4())) - + calibre_db.create_functions(config) meta, error = file_handling_on_upload(requested_file) if error: return error @@ -275,9 +140,7 @@ def upload(): input_authors[0], meta.file_path, title_dir + meta.extension.lower()) - move_coverfile(meta, db_book) - if modify_date: calibre_db.set_metadata_dirty(book_id) # save data to database, reread data @@ -295,16 +158,17 @@ def upload(): if len(request.files.getlist("btn-upload")) < 2: if current_user.role_edit() or current_user.role_admin(): resp = {"location": url_for('edit-book.show_edit_book', book_id=book_id)} - return Response(json.dumps(resp), mimetype='application/json') + return make_response(jsonify(resp)) else: resp = {"location": url_for('web.show_book', book_id=book_id)} - return Response(json.dumps(resp), mimetype='application/json') + return make_response(jsonify(resp)) except (OperationalError, IntegrityError, StaleDataError) as e: calibre_db.session.rollback() log.error_or_exception("Database error: {}".format(e)) flash(_("Oops! Database Error: %(error)s.", error=e.orig if hasattr(e, "orig") else e), category="error") - return Response(json.dumps({"location": url_for("web.index")}), mimetype='application/json') + return make_response(jsonify(location=url_for("web.index"))) + abort(404) @editbook.route("/admin/book/convert/", methods=['POST']) @@ -342,7 +206,7 @@ def table_get_custom_enum(c_id): ret.append({'value': "", 'text': ""}) for idx, en in enumerate(cc.get_display_dict()['enum_values']): ret.append({'value': en, 'text': en}) - return json.dumps(ret) + return make_response(jsonify(ret)) @editbook.route("/ajax/editbooks/", methods=['POST']) @@ -351,73 +215,60 @@ def table_get_custom_enum(c_id): def edit_list_book(param): vals = request.form.to_dict() book = calibre_db.get_book(vals['pk']) + calibre_db.create_functions(config) sort_param = "" ret = "" try: if param == 'series_index': edit_book_series_index(vals['value'], book) - ret = Response(json.dumps({'success': True, 'newValue': book.series_index}), mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=book.series_index)) elif param == 'tags': edit_book_tags(vals['value'], book) - ret = Response(json.dumps({'success': True, 'newValue': ', '.join([tag.name for tag in book.tags])}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=', '.join([tag.name for tag in book.tags]))) elif param == 'series': edit_book_series(vals['value'], book) - ret = Response(json.dumps({'success': True, 'newValue': ', '.join([serie.name for serie in book.series])}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=', '.join([serie.name for serie in book.series]))) elif param == 'publishers': edit_book_publisher(vals['value'], book) - ret = Response(json.dumps({'success': True, - 'newValue': ', '.join([publisher.name for publisher in book.publishers])}), - mimetype='application/json') + ret = make_response(jsonify(success=True, + newValue=', '.join([publisher.name for publisher in book.publishers]))) elif param == 'languages': invalid = list() edit_book_languages(vals['value'], book, invalid=invalid) if invalid: - ret = Response(json.dumps({'success': False, - 'msg': 'Invalid languages in request: {}'.format(','.join(invalid))}), - mimetype='application/json') + ret = make_response(jsonify(success=False, + msg='Invalid languages in request: {}'.format(','.join(invalid)))) else: lang_names = list() for lang in book.languages: lang_names.append(isoLanguages.get_language_name(get_locale(), lang.lang_code)) - ret = Response(json.dumps({'success': True, 'newValue': ', '.join(lang_names)}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=', '.join(lang_names))) elif param == 'author_sort': book.author_sort = vals['value'] - ret = Response(json.dumps({'success': True, 'newValue': book.author_sort}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=book.author_sort)) elif param == 'title': sort_param = book.sort if handle_title_on_edit(book, vals.get('value', "")): rename_error = helper.update_dir_structure(book.id, config.get_book_path()) if not rename_error: - ret = Response(json.dumps({'success': True, 'newValue': book.title}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=book.title)) else: - ret = Response(json.dumps({'success': False, - 'msg': rename_error}), - mimetype='application/json') + ret = make_response(jsonify(success=False, msg=rename_error)) elif param == 'sort': book.sort = vals['value'] - ret = Response(json.dumps({'success': True, 'newValue': book.sort}), - mimetype='application/json') + ret = make_response(jsonify(success=True,newValue=book.sort)) elif param == 'comments': edit_book_comments(vals['value'], book) - ret = Response(json.dumps({'success': True, 'newValue': book.comments[0].text}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=book.comments[0].text)) elif param == 'authors': input_authors, __ = handle_author_on_edit(book, vals['value'], vals.get('checkA', None) == "true") rename_error = helper.update_dir_structure(book.id, config.get_book_path(), input_authors[0]) if not rename_error: - ret = Response(json.dumps({ - 'success': True, - 'newValue': ' & '.join([author.replace('|', ',') for author in input_authors])}), - mimetype='application/json') + ret = make_response(jsonify( + success=True, + newValue=' & '.join([author.replace('|', ',') for author in input_authors]))) else: - ret = Response(json.dumps({'success': False, - 'msg': rename_error}), - mimetype='application/json') + ret = make_response(jsonify(success=False, msg=rename_error)) elif param == 'is_archived': is_archived = change_archived_books(book.id, vals['value'] == "True", message="Book {} archive bit set to: {}".format(book.id, vals['value'])) @@ -436,11 +287,10 @@ def edit_list_book(param): if vals['value'] in ["True", "False"]: ret = "" else: - ret = Response(json.dumps({'success': True, 'newValue': vals['value']}), - mimetype='application/json') + ret = make_response(jsonify(success=True, newValue=vals['value'])) else: return _("Parameter not found"), 400 - book.last_modified = datetime.utcnow() + book.last_modified = datetime.now(timezone.utc) calibre_db.session.commit() # revert change for sort if automatic fields link is deactivated @@ -450,9 +300,8 @@ def edit_list_book(param): except (OperationalError, IntegrityError, StaleDataError) as e: calibre_db.session.rollback() log.error_or_exception("Database error: {}".format(e)) - ret = Response(json.dumps({'success': False, - 'msg': 'Database error: {}'.format(e.orig if hasattr(e, "orig") else e)}), - mimetype='application/json') + ret = make_response(jsonify(success=False, + msg='Database error: {}'.format(e.orig if hasattr(e, "orig") else e))) return ret @@ -463,13 +312,13 @@ def get_sorted_entry(field, bookid): book = calibre_db.get_filtered_book(bookid) if book: if field == 'title': - return json.dumps({'sort': book.sort}) + return make_response(jsonify(sort=book.sort)) elif field == 'authors': - return json.dumps({'author_sort': book.author_sort}) + return make_response(jsonify(author_sort=book.author_sort)) if field == 'sort': - return json.dumps({'sort': book.title}) + return make_response(jsonify(sort=book.title)) if field == 'author_sort': - return json.dumps({'authors': " & ".join([a.name for a in calibre_db.order_authors([book])])}) + return make_response(jsonify(authors=" & ".join([a.name for a in calibre_db.order_authors([book])]))) return "" @@ -485,7 +334,7 @@ def simulate_merge_list_book(): from_book = [] for book_id in vals: from_book.append(calibre_db.get_book(book_id).title) - return json.dumps({'to': to_book, 'from': from_book}) + return make_response(jsonify({'to': to_book, 'from': from_book})) return "" @@ -523,7 +372,7 @@ def merge_list_book(): element.uncompressed_size, to_name)) delete_book_from_table(from_book.id, "", True) - return json.dumps({'success': True}) + return make_response(jsonify(success=True)) return "" @@ -556,38 +405,182 @@ def table_xchange_author_title(): # toDo: Handle error edit_error = helper.update_dir_structure(edited_books_id, config.get_book_path(), input_authors[0]) if modify_date: - book.last_modified = datetime.utcnow() + book.last_modified = datetime.now(timezone.utc) calibre_db.set_metadata_dirty(book.id) try: calibre_db.session.commit() except (OperationalError, IntegrityError, StaleDataError) as e: calibre_db.session.rollback() log.error_or_exception("Database error: {}".format(e)) - return json.dumps({'success': False}) + return make_response(jsonify(success=False)) if config.config_use_google_drive: gdriveutils.updateGdriveCalibreFromLocal() - return json.dumps({'success': True}) + return make_response(jsonify(success=True)) return "" -def merge_metadata(to_save, meta): - if to_save.get('author_name', "") == _('Unknown'): - to_save['author_name'] = '' - if to_save.get('book_title', "") == _('Unknown'): - to_save['book_title'] = '' - if not to_save["languages"] and meta.languages: - upload_language = True - else: - upload_language = False +def do_edit_book(book_id, upload_formats=None): + modify_date = False + edit_error = False + + # create the function for sorting... + calibre_db.create_functions(config) + + book = calibre_db.get_filtered_book(book_id, allow_show_archived=True) + # Book not found + if not book: + flash(_("Oops! Selected book is unavailable. File does not exist or is not accessible"), + category="error") + return redirect(url_for("web.index")) + + to_save = request.form.to_dict() + + try: + # Update folder of book on local disk + edited_books_id = None + title_author_error = None + # upload_mode = False + # handle book title change + if "title" in to_save: + title_change = handle_title_on_edit(book, to_save["title"]) + # handle book author change + if not upload_formats: + input_authors, author_change = handle_author_on_edit(book, to_save["authors"]) + if author_change or title_change: + edited_books_id = book.id + modify_date = True + title_author_error = helper.update_dir_structure(edited_books_id, + config.get_book_path(), + input_authors[0]) + if title_author_error: + flash(title_author_error, category="error") + calibre_db.session.rollback() + book = calibre_db.get_filtered_book(book_id, allow_show_archived=True) + + # handle book ratings + modify_date |= edit_book_ratings(to_save, book) + else: + # handle upload other formats from local disk + to_save, edit_error = upload_book_formats(upload_formats, book, book_id, book.has_cover) + # handle upload covers from local disk + cover_upload_success = upload_cover(request, book) + if cover_upload_success or to_save.get("format_cover"): + book.has_cover = 1 + modify_date = True + + # upload new covers or new file formats to google drive + if config.config_use_google_drive: + gdriveutils.updateGdriveCalibreFromLocal() + + if to_save.get("cover_url",): + if not current_user.role_upload(): + edit_error = True + flash(_("User has no rights to upload cover"), category="error") + if to_save["cover_url"].endswith('/static/generic_cover.jpg'): + book.has_cover = 0 + else: + result, error = helper.save_cover_from_url(to_save["cover_url"].strip(), book.path) + if result is True: + book.has_cover = 1 + modify_date = True + helper.replace_cover_thumbnail_cache(book.id) + else: + edit_error = True + flash(error, category="error") + + # Add default series_index to book + modify_date |= edit_book_series_index(to_save.get("series_index"), book) + # Handle book comments/description + modify_date |= edit_book_comments(Markup(to_save.get('comments')).unescape(), book) + # Handle identifiers + input_identifiers = identifier_list(to_save, book) + modification, warning = modify_identifiers(input_identifiers, book.identifiers, calibre_db.session) + if warning: + flash(_("Identifiers are not Case Sensitive, Overwriting Old Identifier"), category="warning") + modify_date |= modification + # Handle book tags + modify_date |= edit_book_tags(to_save.get('tags'), book) + # Handle book series + modify_date |= edit_book_series(to_save.get("series"), book) + # handle book publisher + modify_date |= edit_book_publisher(to_save.get('publisher'), book) + # handle book languages + try: + invalid = [] + modify_date |= edit_book_languages(to_save.get('languages'), book, upload_mode=upload_formats, + invalid=invalid) + if invalid: + for lang in invalid: + flash(_("'%(langname)s' is not a valid language", langname=lang), category="warning") + except ValueError as e: + flash(str(e), category="error") + edit_error = True + # handle cc data + modify_date |= edit_all_cc_data(book_id, book, to_save) + + if to_save.get("pubdate") is not None: + if to_save.get("pubdate"): + try: + book.pubdate = datetime.strptime(to_save["pubdate"], "%Y-%m-%d") + except ValueError as e: + book.pubdate = db.Books.DEFAULT_PUBDATE + flash(str(e), category="error") + edit_error = True + else: + book.pubdate = db.Books.DEFAULT_PUBDATE + + if modify_date: + book.last_modified = datetime.now(timezone.utc) + kobo_sync_status.remove_synced_book(edited_books_id, all=True) + calibre_db.set_metadata_dirty(book.id) + + calibre_db.session.merge(book) + calibre_db.session.commit() + if config.config_use_google_drive: + gdriveutils.updateGdriveCalibreFromLocal() + if edit_error is not True and title_author_error is not True and cover_upload_success is not False: + flash(_("Metadata successfully updated"), category="success") + + if upload_formats: + resp = {"location": url_for('edit-book.show_edit_book', book_id=book_id)} + return make_response(jsonify(resp)) + + if "detail_view" in to_save: + return redirect(url_for('web.show_book', book_id=book.id)) + else: + return render_edit_book(book_id) + except ValueError as e: + log.error_or_exception("Error: {}".format(e)) + calibre_db.session.rollback() + flash(str(e), category="error") + return redirect(url_for('web.show_book', book_id=book.id)) + except (OperationalError, IntegrityError, StaleDataError, InterfaceError) as e: + log.error_or_exception("Database error: {}".format(e)) + calibre_db.session.rollback() + flash(_("Oops! Database Error: %(error)s.", error=e.orig if hasattr(e, "orig") else e), category="error") + return redirect(url_for('web.show_book', book_id=book.id)) + except Exception as ex: + log.error_or_exception(ex) + calibre_db.session.rollback() + flash(_("Error editing book: {}".format(ex)), category="error") + return redirect(url_for('web.show_book', book_id=book.id)) + + +def merge_metadata(book, meta, to_save): + if meta.cover: + to_save['cover_format'] = meta.cover for s_field, m_field in [ - ('tags', 'tags'), ('author_name', 'author'), ('series', 'series'), + ('tags', 'tags'), ('authors', 'author'), ('series', 'series'), ('series_index', 'series_id'), ('languages', 'languages'), - ('book_title', 'title')]: - to_save[s_field] = to_save[s_field] or getattr(meta, m_field, '') - to_save["description"] = to_save["description"] or Markup( - getattr(meta, 'description', '')).unescape() - return upload_language + ('title', 'title'), ('comments', 'description')]: + try: + val = None if len(getattr(book, s_field)) else getattr(meta, m_field, '') + except TypeError: + val = None if len(str(getattr(book, s_field))) else getattr(meta, m_field, '') + if val: + to_save[s_field] = val + def identifier_list(to_save, book): """Generate a list of Identifiers from form information""" @@ -707,8 +700,8 @@ def create_book_on_upload(modify_date, meta): pubdate = datetime(101, 1, 1) # Calibre adds books with utc as timezone - db_book = db.Books(title, "", sort_authors, datetime.utcnow(), pubdate, - '1', datetime.utcnow(), path, meta.cover, db_author, [], "") + db_book = db.Books(title, "", sort_authors, datetime.now(timezone.utc), pubdate, + '1', datetime.now(timezone.utc), path, meta.cover, db_author, [], "") modify_date |= modify_database_object(input_authors, db_book.authors, db.Authors, calibre_db.session, 'author') @@ -760,17 +753,17 @@ def file_handling_on_upload(requested_file): if config.config_check_extensions and allowed_extensions != ['']: if not validate_mime_type(requested_file, allowed_extensions): flash(_("File type isn't allowed to be uploaded to this server"), category="error") - return None, Response(json.dumps({"location": url_for("web.index")}), mimetype='application/json') + return None, make_response(jsonify(location=url_for("web.index"))) if '.' in requested_file.filename: file_ext = requested_file.filename.rsplit('.', 1)[-1].lower() if file_ext not in allowed_extensions and '' not in allowed_extensions: flash( _("File extension '%(ext)s' is not allowed to be uploaded to this server", ext=file_ext), category="error") - return None, Response(json.dumps({"location": url_for("web.index")}), mimetype='application/json') + return None, make_response(jsonify(location=url_for("web.index"))) else: flash(_('File to be uploaded must have an extension'), category="error") - return None, Response(json.dumps({"location": url_for("web.index")}), mimetype='application/json') + return None, make_response(jsonify(location=url_for("web.index"))) # extract metadata from file try: @@ -779,7 +772,7 @@ def file_handling_on_upload(requested_file): log.error("File %s could not saved to temp dir", requested_file.filename) flash(_("File %(filename)s could not saved to temp dir", filename=requested_file.filename), category="error") - return None, Response(json.dumps({"location": url_for("web.index")}), mimetype='application/json') + return None, make_response(jsonify(location=url_for("web.index"))) return meta, None @@ -851,7 +844,7 @@ def delete_whole_book(book_id, book): def render_delete_book_result(book_format, json_response, warning, book_id, location=""): if book_format: if json_response: - return json.dumps([warning, {"location": url_for("edit-book.show_edit_book", book_id=book_id), + return jsonify([warning, {"location": url_for("edit-book.show_edit_book", book_id=book_id), "type": "success", "format": book_format, "message": _('Book Format Successfully Deleted')}]) @@ -860,7 +853,7 @@ def render_delete_book_result(book_format, json_response, warning, book_id, loca return redirect(url_for('edit-book.show_edit_book', book_id=book_id)) else: if json_response: - return json.dumps([warning, {"location": get_redirect_location(location, "web.index"), + return jsonify([warning, {"location": get_redirect_location(location, "web.index"), "type": "success", "format": book_format, "message": _('Book Successfully Deleted')}]) @@ -878,7 +871,7 @@ def delete_book_from_table(book_id, book_format, json_response, location=""): result, error = helper.delete_book(book, config.get_book_path(), book_format=book_format.upper()) if not result: if json_response: - return json.dumps([{"location": url_for("edit-book.show_edit_book", book_id=book_id), + return jsonify([{"location": url_for("edit-book.show_edit_book", book_id=book_id), "type": "danger", "format": "", "message": error}]) @@ -905,7 +898,7 @@ def delete_book_from_table(book_id, book_format, json_response, location=""): log.error_or_exception(ex) calibre_db.session.rollback() if json_response: - return json.dumps([{"location": url_for("edit-book.show_edit_book", book_id=book_id), + return jsonify([{"location": url_for("edit-book.show_edit_book", book_id=book_id), "type": "danger", "format": "", "message": ex}]) @@ -919,7 +912,7 @@ def delete_book_from_table(book_id, book_format, json_response, location=""): return render_delete_book_result(book_format, json_response, warning, book_id, location) message = _("You are missing permissions to delete books") if json_response: - return json.dumps({"location": url_for("edit-book.show_edit_book", book_id=book_id), + return jsonify({"location": url_for("edit-book.show_edit_book", book_id=book_id), "type": "danger", "format": "", "message": message}) @@ -975,7 +968,7 @@ def render_edit_book(book_id): def edit_book_ratings(to_save, book): changed = False - if to_save.get("rating", "").strip(): + if strip_whitespaces(to_save.get("rating", "")): old_rating = False if len(book.ratings) > 0: old_rating = book.ratings[0].rating @@ -998,84 +991,93 @@ def edit_book_ratings(to_save, book): def edit_book_tags(tags, book): - input_tags = tags.split(',') - input_tags = list(map(lambda it: it.strip(), input_tags)) - # Remove duplicates - input_tags = helper.uniq(input_tags) - return modify_database_object(input_tags, book.tags, db.Tags, calibre_db.session, 'tags') - + if tags is not None: + input_tags = tags.split(',') + input_tags = list(map(lambda it: strip_whitespaces(it), input_tags)) + # Remove duplicates + input_tags = helper.uniq(input_tags) + return modify_database_object(input_tags, book.tags, db.Tags, calibre_db.session, 'tags') + return False def edit_book_series(series, book): - input_series = [series.strip()] - input_series = [x for x in input_series if x != ''] - return modify_database_object(input_series, book.series, db.Series, calibre_db.session, 'series') + if series is not None: + input_series = [strip_whitespaces(series)] + input_series = [x for x in input_series if x != ''] + return modify_database_object(input_series, book.series, db.Series, calibre_db.session, 'series') + return False def edit_book_series_index(series_index, book): - # Add default series_index to book - modify_date = False - series_index = series_index or '1' - if not series_index.replace('.', '', 1).isdigit(): - flash(_("%(seriesindex)s is not a valid number, skipping", seriesindex=series_index), category="warning") - return False - if str(book.series_index) != series_index: - book.series_index = series_index - modify_date = True - return modify_date + if series_index: + # Add default series_index to book + modify_date = False + series_index = series_index or '1' + if not series_index.replace('.', '', 1).isdigit(): + flash(_("Seriesindex: %(seriesindex)s is not a valid number, skipping", seriesindex=series_index), category="warning") + return False + if str(book.series_index) != series_index: + book.series_index = series_index + modify_date = True + return modify_date + return False # Handle book comments/description def edit_book_comments(comments, book): - modify_date = False - if comments: - comments = clean_string(comments, book.id) - if len(book.comments): - if book.comments[0].text != comments: - book.comments[0].text = comments - modify_date = True - else: + if comments is not None: + modify_date = False if comments: - book.comments.append(db.Comments(comment=comments, book=book.id)) - modify_date = True - return modify_date + comments = clean_string(comments, book.id) + if len(book.comments): + if book.comments[0].text != comments: + book.comments[0].text = comments + modify_date = True + else: + if comments: + book.comments.append(db.Comments(comment=comments, book=book.id)) + modify_date = True + return modify_date def edit_book_languages(languages, book, upload_mode=False, invalid=None): - input_languages = languages.split(',') - unknown_languages = [] - if not upload_mode: - input_l = isoLanguages.get_language_codes(get_locale(), input_languages, unknown_languages) - else: - input_l = isoLanguages.get_valid_language_codes(get_locale(), input_languages, unknown_languages) - for lang in unknown_languages: - log.error("'%s' is not a valid language", lang) - if isinstance(invalid, list): - invalid.append(lang) + if languages is not None: + input_languages = languages.split(',') + unknown_languages = [] + if not upload_mode: + input_l = isoLanguages.get_language_code_from_name(get_locale(), input_languages, unknown_languages) else: - raise ValueError(_("'%(langname)s' is not a valid language", langname=lang)) - # ToDo: Not working correct - if upload_mode and len(input_l) == 1: - # If the language of the file is excluded from the users view, it's not imported, to allow the user to view - # the book it's language is set to the filter language - if input_l[0] != current_user.filter_language() and current_user.filter_language() != "all": - input_l[0] = calibre_db.session.query(db.Languages). \ - filter(db.Languages.lang_code == current_user.filter_language()).first().lang_code - # Remove duplicates - input_l = helper.uniq(input_l) - return modify_database_object(input_l, book.languages, db.Languages, calibre_db.session, 'languages') + input_l = isoLanguages.get_valid_language_codes_from_code(get_locale(), input_languages, unknown_languages) + for lang in unknown_languages: + log.error("'%s' is not a valid language", lang) + if isinstance(invalid, list): + invalid.append(lang) + else: + raise ValueError(_("'%(langname)s' is not a valid language", langname=lang)) + # ToDo: Not working correct + if upload_mode and len(input_l) == 1: + # If the language of the file is excluded from the users view, it's not imported, to allow the user to view + # the book it's language is set to the filter language + if input_l[0] != current_user.filter_language() and current_user.filter_language() != "all": + input_l[0] = calibre_db.session.query(db.Languages). \ + filter(db.Languages.lang_code == current_user.filter_language()).first().lang_code + # Remove duplicates from normalized langcodes + input_l = helper.uniq(input_l) + return modify_database_object(input_l, book.languages, db.Languages, calibre_db.session, 'languages') + return False def edit_book_publisher(publishers, book): - changed = False - if publishers: - publisher = publishers.rstrip().strip() - if len(book.publishers) == 0 or (len(book.publishers) > 0 and publisher != book.publishers[0].name): - changed |= modify_database_object([publisher], book.publishers, db.Publishers, calibre_db.session, - 'publisher') - elif len(book.publishers): - changed |= modify_database_object([], book.publishers, db.Publishers, calibre_db.session, 'publisher') - return changed - + if publishers is not None: + changed = False + if publishers: + publisher = strip_whitespaces(publishers) + if len(book.publishers) == 0 or (len(book.publishers) > 0 and publisher != book.publishers[0].name): + changed |= modify_database_object([publisher], book.publishers, db.Publishers, calibre_db.session, + 'publisher') + elif len(book.publishers): + changed |= modify_database_object([], book.publishers, db.Publishers, calibre_db.session, 'publisher') + return changed + return False def edit_cc_data_value(book_id, book, c, to_save, cc_db_value, cc_string): changed = False @@ -1115,7 +1117,7 @@ def edit_cc_data_string(book, c, to_save, cc_db_value, cc_string): changed = False if c.datatype == 'rating': to_save[cc_string] = str(int(float(to_save[cc_string]) * 2)) - if to_save[cc_string].strip() != cc_db_value: + if strip_whitespaces(to_save[cc_string]) != cc_db_value: if cc_db_value is not None: # remove old cc_val del_cc = getattr(book, cc_string)[0] @@ -1125,15 +1127,15 @@ def edit_cc_data_string(book, c, to_save, cc_db_value, cc_string): changed = True cc_class = db.cc_classes[c.id] new_cc = calibre_db.session.query(cc_class).filter( - cc_class.value == to_save[cc_string].strip()).first() + cc_class.value == strip_whitespaces(to_save[cc_string])).first() # if no cc val is found add it if new_cc is None: - new_cc = cc_class(value=to_save[cc_string].strip()) + new_cc = cc_class(value=strip_whitespaces(to_save[cc_string])) calibre_db.session.add(new_cc) changed = True calibre_db.session.flush() new_cc = calibre_db.session.query(cc_class).filter( - cc_class.value == to_save[cc_string].strip()).first() + cc_class.value == strip_whitespaces(to_save[cc_string])).first() # add cc value to book getattr(book, cc_string).append(new_cc) return changed, to_save @@ -1156,61 +1158,66 @@ def edit_cc_data(book_id, book, to_save, cc): changed = False for c in cc: cc_string = "custom_column_" + str(c.id) - if not c.is_multiple: - if len(getattr(book, cc_string)) > 0: - cc_db_value = getattr(book, cc_string)[0].value - else: - cc_db_value = None - if to_save[cc_string].strip(): - if c.datatype in ['int', 'bool', 'float', "datetime", "comments"]: - change, to_save = edit_cc_data_value(book_id, book, c, to_save, cc_db_value, cc_string) + if to_save.get(cc_string) is not None: + if not c.is_multiple: + if len(getattr(book, cc_string)) > 0: + cc_db_value = getattr(book, cc_string)[0].value else: - change, to_save = edit_cc_data_string(book, c, to_save, cc_db_value, cc_string) - changed |= change + cc_db_value = None + if strip_whitespaces(to_save[cc_string]): + if c.datatype in ['int', 'bool', 'float', "datetime", "comments"]: + change, to_save = edit_cc_data_value(book_id, book, c, to_save, cc_db_value, cc_string) + else: + change, to_save = edit_cc_data_string(book, c, to_save, cc_db_value, cc_string) + changed |= change + else: + if cc_db_value is not None: + # remove old cc_val + del_cc = getattr(book, cc_string)[0] + getattr(book, cc_string).remove(del_cc) + if not del_cc.books or len(del_cc.books) == 0: + calibre_db.session.delete(del_cc) + changed = True else: - if cc_db_value is not None: - # remove old cc_val - del_cc = getattr(book, cc_string)[0] - getattr(book, cc_string).remove(del_cc) - if not del_cc.books or len(del_cc.books) == 0: - calibre_db.session.delete(del_cc) - changed = True - else: - input_tags = to_save[cc_string].split(',') - input_tags = list(map(lambda it: it.strip(), input_tags)) - changed |= modify_database_object(input_tags, - getattr(book, cc_string), - db.cc_classes[c.id], - calibre_db.session, - 'custom') + input_tags = to_save[cc_string].split(',') + input_tags = list(map(lambda it: strip_whitespaces(it), input_tags)) + changed |= modify_database_object(input_tags, + getattr(book, cc_string), + db.cc_classes[c.id], + calibre_db.session, + 'custom') return changed -# returns None if no file is uploaded -# returns False if an error occurs, in all other cases the ebook metadata is returned -def upload_single_file(file_request, book, book_id): +# returns False if an error occurs or no book is uploaded, in all other cases the ebook metadata to change is returned +def upload_book_formats(requested_files, book, book_id, no_cover=True): # Check and handle Uploaded file - requested_file = file_request.files.get('btn-upload-format', None) + to_save = dict() + error = False allowed_extensions = config.config_upload_formats.split(',') - if requested_file: + for requested_file in requested_files: + current_filename = requested_file.filename if config.config_check_extensions and allowed_extensions != ['']: if not validate_mime_type(requested_file, allowed_extensions): flash(_("File type isn't allowed to be uploaded to this server"), category="error") - return False - # check for empty request - if requested_file.filename != '': + error = True + continue + if current_filename != '': if not current_user.role_upload(): flash(_("User has no rights to upload additional file formats"), category="error") - return False - if '.' in requested_file.filename: - file_ext = requested_file.filename.rsplit('.', 1)[-1].lower() + error = True + continue + if '.' in current_filename: + file_ext = current_filename.rsplit('.', 1)[-1].lower() if file_ext not in allowed_extensions and '' not in allowed_extensions: flash(_("File extension '%(ext)s' is not allowed to be uploaded to this server", ext=file_ext), category="error") - return False + error = True + continue else: flash(_('File to be uploaded must have an extension'), category="error") - return False + error = True + continue file_name = book.path.rsplit('/', 1)[-1] filepath = os.path.normpath(os.path.join(config.get_book_path(), book.path)) @@ -1223,41 +1230,50 @@ def upload_single_file(file_request, book, book_id): except OSError: flash(_("Failed to create path %(path)s (Permission denied).", path=filepath), category="error") - return False + error = True + continue try: requested_file.save(saved_filename) except OSError: flash(_("Failed to store file %(file)s.", file=saved_filename), category="error") - return False + error = True + continue file_size = os.path.getsize(saved_filename) - is_format = calibre_db.get_book_format(book_id, file_ext.upper()) # Format entry already exists, no need to update the database - if is_format: + if calibre_db.get_book_format(book_id, file_ext.upper()): log.warning('Book format %s already existing', file_ext.upper()) else: try: db_format = db.Data(book_id, file_ext.upper(), file_size, file_name) calibre_db.session.add(db_format) calibre_db.session.commit() - calibre_db.update_title_sort(config) + calibre_db.create_functions(config) except (OperationalError, IntegrityError, StaleDataError) as e: calibre_db.session.rollback() log.error_or_exception("Database error: {}".format(e)) flash(_("Oops! Database Error: %(error)s.", error=e.orig if hasattr(e, "orig") else e), category="error") - return False # return redirect(url_for('web.show_book', book_id=book.id)) + error = True + continue # Queue uploader info link = '{}'.format(url_for('web.show_book', book_id=book.id), escape(book.title)) upload_text = N_("File format %(ext)s added to %(book)s", ext=file_ext.upper(), book=link) WorkerThread.add(current_user.name, TaskUpload(upload_text, escape(book.title))) - - return uploader.process( - saved_filename, *os.path.splitext(requested_file.filename), - rar_executable=config.config_rarfile_location) - return None + meta = uploader.process( + saved_filename, + *os.path.splitext(current_filename), + rar_executable=config.config_rarfile_location, + no_cover=no_cover) + merge_metadata(book, meta, to_save) + #if to_save.get('languages'): + # langs = [] + # for lang_code in to_save['languages'].split(','): + # langs.append(isoLanguages.get_language_name(get_locale(), lang_code)) + # to_save['languages'] = ",".join(langs) + return to_save, error def upload_cover(cover_request, book): @@ -1280,7 +1296,7 @@ def upload_cover(cover_request, book): def handle_title_on_edit(book, book_title): # handle book title - book_title = book_title.rstrip().strip() + book_title = strip_whitespaces(book_title) if book.title != book_title: if book_title == '': book_title = _(u'Unknown') @@ -1291,7 +1307,6 @@ def handle_title_on_edit(book, book_title): def handle_author_on_edit(book, author_name, update_stored=True): change = False - # handle author(s) input_authors = prepare_authors(author_name, config.get_book_path(), config.config_use_google_drive) # Search for each author if author is in database, if not, author name and sorted author name is generated new @@ -1321,7 +1336,6 @@ def search_objects_remove(db_book_object, db_type, input_elements): if db_type == 'custom': type_elements = c_elements.value else: - # type_elements = c_elements.name type_elements = c_elements for inp_element in input_elements: if type_elements == inp_element: diff --git a/cps/embed_helper.py b/cps/embed_helper.py index 54db47ef..f222bcd0 100644 --- a/cps/embed_helper.py +++ b/cps/embed_helper.py @@ -35,9 +35,7 @@ def do_calibre_export(book_id, book_format): my_env = os.environ.copy() if config.config_calibre_split: my_env['CALIBRE_OVERRIDE_DATABASE_PATH'] = os.path.join(config.config_calibre_dir, "metadata.db") - library_path = config.config_calibre_split_dir - else: - library_path = config.config_calibre_dir + library_path = config.get_book_path() opf_command = [calibredb_binarypath, 'export', '--dont-write-opf', '--with-library', library_path, '--to-dir', tmp_dir, '--formats', book_format, "--template", "{}".format(temp_file_name), str(book_id)] diff --git a/cps/epub.py b/cps/epub.py index c802f61d..3947cee6 100644 --- a/cps/epub.py +++ b/cps/epub.py @@ -25,6 +25,7 @@ from . import config, logger from .helper import split_authors from .epub_helper import get_content_opf, default_ns from .constants import BookMeta +from .string_helper import strip_whitespaces log = logger.create() @@ -55,7 +56,7 @@ def get_epub_layout(book, book_data): p = tree.xpath('/pkg:package/pkg:metadata', namespaces=default_ns)[0] layout = p.xpath('pkg:meta[@property="rendition:layout"]/text()', namespaces=default_ns) - except (etree.XMLSyntaxError, KeyError, IndexError, OSError) as e: + except (etree.XMLSyntaxError, KeyError, IndexError, OSError, UnicodeDecodeError) as e: log.error("Could not parse epub metadata of book {} during kobo sync: {}".format(book.id, e)) layout = [] @@ -65,7 +66,7 @@ def get_epub_layout(book, book_data): return layout[0] -def get_epub_info(tmp_file_path, original_file_name, original_file_extension): +def get_epub_info(tmp_file_path, original_file_name, original_file_extension, no_cover_processing): ns = { 'n': 'urn:oasis:names:tc:opendocument:xmlns:container', 'pkg': 'http://www.idpf.org/2007/opf', @@ -90,7 +91,7 @@ def get_epub_info(tmp_file_path, original_file_name, original_file_extension): elif s == 'date': epub_metadata[s] = tmp[0][:10] else: - epub_metadata[s] = tmp[0].strip() + epub_metadata[s] = strip_whitespaces(tmp[0]) else: epub_metadata[s] = 'Unknown' @@ -116,7 +117,10 @@ def get_epub_info(tmp_file_path, original_file_name, original_file_extension): epub_metadata = parse_epub_series(ns, tree, epub_metadata) epub_zip = zipfile.ZipFile(tmp_file_path) - cover_file = parse_epub_cover(ns, tree, epub_zip, cover_path, tmp_file_path) + if not no_cover_processing: + cover_file = parse_epub_cover(ns, tree, epub_zip, cover_path, tmp_file_path) + else: + cover_file = None identifiers = [] for node in p.xpath('dc:identifier', namespaces=ns): diff --git a/cps/error_handler.py b/cps/error_handler.py index 92d3e876..0df98332 100644 --- a/cps/error_handler.py +++ b/cps/error_handler.py @@ -37,17 +37,31 @@ def error_http(error): error_code="Error {0}".format(error.code), error_name=error.name, issue=False, + goto_admin=False, unconfigured=not config.db_configured, instance=config.config_calibre_web_title ), error.code def internal_error(error): + if (isinstance(error.original_exception, AttributeError) and + error.original_exception.args[0] == "'NoneType' object has no attribute 'query'" + and error.original_exception.name == "query"): + return render_template('http_error.html', + error_code="Database Error", + error_name='The library used is invalid or has permission errors', + issue=False, + goto_admin=True, + unconfigured=False, + error_stack="", + instance=config.config_calibre_web_title + ), 500 return render_template('http_error.html', error_code="500 Internal Server Error", error_name='The server encountered an internal error and was unable to complete your ' 'request. There is an error in the application.', issue=True, + goto_admin=False, unconfigured=False, error_stack=traceback.format_exc().split("\n"), instance=config.config_calibre_web_title diff --git a/cps/file_helper.py b/cps/file_helper.py index 9d5406d3..361709f8 100644 --- a/cps/file_helper.py +++ b/cps/file_helper.py @@ -34,6 +34,15 @@ except ImportError as e: error = "Cannot import python-magic, checking uploaded file metadata will not work: {}".format(e) +def get_mimetype(ext): + # overwrite some mimetypes for proper file detection + mimes = {".fb2": "text/xml", + ".cbz": "application/zip", + ".cbr": "application/x-rar" + } + return mimes.get(ext, mimetypes.types_map[ext]) + + def get_temp_dir(): tmp_dir = os.path.join(gettempdir(), 'calibre_web') if not os.path.isdir(tmp_dir): @@ -54,7 +63,7 @@ def validate_mime_type(file_buffer, allowed_extensions): allowed_mimetypes = list() for x in allowed_extensions: try: - allowed_mimetypes.append(mimetypes.types_map["." + x]) + allowed_mimetypes.append(get_mimetype("." + x)) except KeyError: log.error("Unkown mimetype for Extension: {}".format(x)) tmp_mime_type = mime.from_buffer(file_buffer.read()) diff --git a/cps/helper.py b/cps/helper.py index 004e1b0e..8fa9bbac 100644 --- a/cps/helper.py +++ b/cps/helper.py @@ -25,12 +25,12 @@ import re import regex import shutil import socket -from datetime import datetime, timedelta +from datetime import datetime, timedelta, timezone import requests import unidecode from uuid import uuid4 -from flask import send_from_directory, make_response, abort, url_for, Response +from flask import send_from_directory, make_response, abort, url_for, Response, request from flask_babel import gettext as _ from flask_babel import lazy_gettext as N_ from flask_babel import get_locale @@ -43,15 +43,16 @@ from markupsafe import escape from urllib.parse import quote try: - import advocate - from advocate.exceptions import UnacceptableAddressException + from . import cw_advocate + from .cw_advocate.exceptions import UnacceptableAddressException use_advocate = True -except ImportError: +except ImportError as e: use_advocate = False advocate = requests UnacceptableAddressException = MissingSchema = BaseException from . import calibre_db, cli_param +from .string_helper import strip_whitespaces from .tasks.convert import TaskConvert from . import logger, config, db, ub, fs from . import gdriveutils as gd @@ -118,7 +119,7 @@ def convert_book_format(book_id, calibre_path, old_book_format, new_book_format, # Texts are not lazy translated as they are supposed to get send out as is def send_test_mail(ereader_mail, user_name): for email in ereader_mail.split(','): - email = email.strip() + email = strip_whitespaces(email) WorkerThread.add(user_name, TaskEmail(_('Calibre-Web Test Email'), None, None, config.get_mail_settings(), email, N_("Test Email"), _('This Email has been sent via Calibre-Web.'))) @@ -198,7 +199,7 @@ def check_send_to_ereader(entry): # Check if a reader is existing for any of the book formats, if not, return empty list, otherwise return # list with supported formats def check_read_formats(entry): - extensions_reader = {'TXT', 'PDF', 'EPUB', 'CBZ', 'CBT', 'CBR', 'DJVU', 'DJV'} + extensions_reader = {'TXT', 'PDF', 'EPUB', 'KEPUB', 'CBZ', 'CBT', 'CBR', 'DJVU', 'DJV'} book_formats = list() if len(entry.data): for ele in iter(entry.data): @@ -228,7 +229,7 @@ def send_mail(book_id, book_format, convert, ereader_mail, calibrepath, user_id) link = '{}'.format(url_for('web.show_book', book_id=book_id), escape(book.title)) email_text = N_("%(book)s send to eReader", book=link) for email in ereader_mail.split(','): - email = email.strip() + email = strip_whitespaces(email) WorkerThread.add(user_id, TaskEmail(_("Send to eReader"), book.path, converted_file_name, config.get_mail_settings(), email, email_text, _('This Email has been sent via Calibre-Web.'), book.id)) @@ -236,7 +237,7 @@ def send_mail(book_id, book_format, convert, ereader_mail, calibrepath, user_id) return _("The requested file could not be read. Maybe wrong permissions?") -def get_valid_filename(value, replace_whitespace=True, chars=128): +def get_valid_filename(value, replace_whitespace=True, chars=128, force_unidecode=False): """ Returns the given string converted to a string that can be used for a clean filename. Limits num characters to 128 max. @@ -244,7 +245,7 @@ def get_valid_filename(value, replace_whitespace=True, chars=128): if value[-1:] == '.': value = value[:-1]+'_' value = value.replace("/", "_").replace(":", "_").strip('\0') - if config.config_unicode_filename: + if config.config_unicode_filename or force_unidecode: value = (unidecode.unidecode(value)) if replace_whitespace: # *+:\"/<>? are replaced by _ @@ -252,7 +253,7 @@ def get_valid_filename(value, replace_whitespace=True, chars=128): # pipe has to be replaced with comma value = re.sub(r'[|]+', ',', value, flags=re.U) - value = value.encode('utf-8')[:chars].decode('utf-8', errors='ignore').strip() + value = strip_whitespaces(value.encode('utf-8')[:chars].decode('utf-8', errors='ignore')) if not value: raise ValueError("Filename cannot be empty") @@ -267,11 +268,11 @@ def split_authors(values): commas = author.count(',') if commas == 1: author_split = author.split(',') - authors_list.append(author_split[1].strip() + ' ' + author_split[0].strip()) + authors_list.append(strip_whitespaces(author_split[1]) + ' ' + strip_whitespaces(author_split[0])) elif commas > 1: - authors_list.extend([x.strip() for x in author.split(',')]) + authors_list.extend([strip_whitespaces(x) for x in author.split(',')]) else: - authors_list.append(author.strip()) + authors_list.append(strip_whitespaces(author)) return authors_list @@ -327,7 +328,7 @@ def edit_book_read_status(book_id, read_status=None): ub.session_commit("Book {} readbit toggled".format(book_id)) else: try: - calibre_db.update_title_sort(config) + calibre_db.create_functions(config) book = calibre_db.get_filtered_book(book_id) book_read_status = getattr(book, 'custom_column_' + str(config.config_read_column)) if len(book_read_status): @@ -417,8 +418,6 @@ def rename_author_path(first_author, old_author_dir, renamed_author, calibre_pat # Create new_author_dir from parameter or from database # Create new title_dir from database and add id new_authordir = get_valid_filename(first_author, chars=96) - # new_author = calibre_db.session.query(db.Authors).filter(db.Authors.name == renamed_author).first() - # old_author_dir = get_valid_filename(old_author_name, chars=96) new_author_rename_dir = get_valid_filename(renamed_author, chars=96) if gdrive: g_file = gd.getFileFromEbooksFolder(None, old_author_dir) @@ -467,7 +466,6 @@ def update_dir_structure_file(book_id, calibre_path, original_filepath, new_auth db_filename, original_filepath, path) - # old_path = os.path.join(calibre_path, author_dir, new_title_dir).replace('\\', '/') new_path = os.path.join(calibre_path, new_author_dir, new_title_dir).replace('\\', '/') all_new_name = get_valid_filename(local_book.title, chars=42) + ' - ' \ + get_valid_filename(new_author, chars=42) @@ -476,8 +474,6 @@ def update_dir_structure_file(book_id, calibre_path, original_filepath, new_auth if error: return error - - # Rename all files from old names to new names return False @@ -489,7 +485,7 @@ def upload_new_file_gdrive(book_id, first_author, title, title_dir, original_fil title_dir + " (" + str(book_id) + ")") book.path = gdrive_path.replace("\\", "/") gd.uploadFileToEbooksFolder(os.path.join(gdrive_path, file_name).replace("\\", "/"), original_filepath) - return False # rename_files_on_change(first_author, renamed_author, local_book=book, gdrive=True) + return False def update_dir_structure_gdrive(book_id, first_author): @@ -517,24 +513,26 @@ def update_dir_structure_gdrive(book_id, first_author): book.path = new_authordir + '/' + book.path.split('/')[1] gd.updateDatabaseOnEdit(g_file['id'], book.path) else: - return _('File %(file)s not found on Google Drive', file=authordir) # file not found''' + return _('File %(file)s not found on Google Drive', file=authordir) # file not found if titledir != new_titledir or authordir != new_authordir : all_new_name = get_valid_filename(book.title, chars=42) + ' - ' \ + get_valid_filename(new_authordir, chars=42) rename_all_files_on_change(book, book.path, book.path, all_new_name, gdrive=True) # todo: Move filenames on gdrive - # change location in database to new author/title path - # book.path = os.path.join(authordir, new_titledir).replace('\\', '/') return False def move_files_on_change(calibre_path, new_author_dir, new_titledir, localbook, db_filename, original_filepath, path): new_path = os.path.join(calibre_path, new_author_dir, new_titledir) - # new_name = get_valid_filename(localbook.title, chars=96) + ' - ' + new_author_dir try: if original_filepath: if not os.path.isdir(new_path): os.makedirs(new_path) - shutil.move(original_filepath, os.path.join(new_path, db_filename)) + try: + shutil.move(original_filepath, os.path.join(new_path, db_filename)) + except OSError: + log.error("Rename title from {} to {} failed with error, trying to " + "move without metadata".format(path, new_path)) + shutil.move(original_filepath, os.path.join(new_path, db_filename), copy_function=shutil.copy) log.debug("Moving title: %s to %s", original_filepath, new_path) else: # Check new path is not valid path @@ -661,7 +659,7 @@ def check_email(email): def check_username(username): - username = username.strip() + username = strip_whitespaces(username) if ub.session.query(ub.User).filter(func.lower(ub.User.name) == username.lower()).scalar(): log.error("This username is already taken") raise Exception(_("This username is already taken")) @@ -669,16 +667,18 @@ def check_username(username): def valid_email(emails): + valid_emails = [] for email in emails.split(','): - email = email.strip() - # if email is not deleted - if email: - # Regex according to https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/email#validation - if not re.search(r"^[\w.!#$%&'*+\\/=?^_`{|}~-]+@[\w](?:[\w-]{0,61}[\w])?(?:\.[\w](?:[\w-]{0,61}[\w])?)*$", - email): - log.error("Invalid Email address format") - raise Exception(_("Invalid Email address format")) - return email + email = strip_whitespaces(email) + # if email is not deleted + if email: + # Regex according to https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/email#validation + if not re.search(r"^[\w.!#$%&'*+\\/=?^_`{|}~-]+@[\w](?:[\w-]{0,61}[\w])?(?:\.[\w](?:[\w-]{0,61}[\w])?)*$", + email): + log.error("Invalid Email address format for {}".format(email)) + raise Exception(_("Invalid Email address format")) + valid_emails.append(email) + return ",".join(valid_emails) def valid_password(check_password): @@ -788,24 +788,23 @@ def get_book_cover_internal(book, resolution=None): def get_book_cover_thumbnail(book, resolution): if book and book.has_cover: - return ub.session \ - .query(ub.Thumbnail) \ - .filter(ub.Thumbnail.type == THUMBNAIL_TYPE_COVER) \ - .filter(ub.Thumbnail.entity_id == book.id) \ - .filter(ub.Thumbnail.resolution == resolution) \ - .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.utcnow())) \ - .first() + return (ub.session + .query(ub.Thumbnail) + .filter(ub.Thumbnail.type == THUMBNAIL_TYPE_COVER) + .filter(ub.Thumbnail.entity_id == book.id) + .filter(ub.Thumbnail.resolution == resolution) + .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.now(timezone.utc))) + .first()) def get_series_thumbnail_on_failure(series_id, resolution): - book = calibre_db.session \ - .query(db.Books) \ - .join(db.books_series_link) \ - .join(db.Series) \ - .filter(db.Series.id == series_id) \ - .filter(db.Books.has_cover == 1) \ - .first() - + book = (calibre_db.session + .query(db.Books) + .join(db.books_series_link) + .join(db.Series) + .filter(db.Series.id == series_id) + .filter(db.Books.has_cover == 1) + .first()) return get_book_cover_internal(book, resolution=resolution) @@ -827,13 +826,13 @@ def get_series_cover_internal(series_id, resolution=None): def get_series_thumbnail(series_id, resolution): - return ub.session \ - .query(ub.Thumbnail) \ - .filter(ub.Thumbnail.type == THUMBNAIL_TYPE_SERIES) \ - .filter(ub.Thumbnail.entity_id == series_id) \ - .filter(ub.Thumbnail.resolution == resolution) \ - .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.utcnow())) \ - .first() + return (ub.session + .query(ub.Thumbnail) + .filter(ub.Thumbnail.type == THUMBNAIL_TYPE_SERIES) + .filter(ub.Thumbnail.entity_id == series_id) + .filter(ub.Thumbnail.resolution == resolution) + .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.now(timezone.utc))) + .first()) # saves book cover from url @@ -842,7 +841,7 @@ def save_cover_from_url(url, book_path): if cli_param.allow_localhost: img = requests.get(url, timeout=(10, 200), allow_redirects=False) # ToDo: Error Handling elif use_advocate: - img = advocate.get(url, timeout=(10, 200), allow_redirects=False) # ToDo: Error Handling + img = cw_advocate.get(url, timeout=(10, 200), allow_redirects=False) # ToDo: Error Handling else: log.error("python module advocate is not installed but is needed") return False, _("Python module 'advocate' is not installed but is needed for cover uploads") @@ -906,7 +905,7 @@ def save_cover(img, book_path): else: imgc = Image(blob=io.BytesIO(img.content)) imgc.format = 'jpeg' - imgc.transform_colorspace("rgb") + imgc.transform_colorspace("srgb") img = imgc except (BlobError, MissingDelegateError): log.error("Invalid cover file content") @@ -975,7 +974,8 @@ def do_download_file(book, book_format, client, data, headers): # ToDo Check headers parameter for element in headers: response.headers[element[0]] = element[1] - log.info('Downloading file: {}'.format(os.path.join(filename, book_name + "." + book_format))) + log.info('Downloading file: \'%s\' by %s - %s', format(os.path.join(filename, book_name + "." + book_format)), + current_user.name, request.headers.get('X-Forwarded-For', request.remote_addr)) return response @@ -1105,11 +1105,14 @@ def get_download_link(book_id, book_format, client): file_name = book.title if len(book.authors) > 0: file_name = file_name + ' - ' + book.authors[0].name - file_name = get_valid_filename(file_name, replace_whitespace=False) + if client == "kindle": + file_name = get_valid_filename(file_name, replace_whitespace=False, force_unidecode=True) + else: + file_name = quote(get_valid_filename(file_name, replace_whitespace=False)) headers = Headers() headers["Content-Type"] = mimetypes.types_map.get('.' + book_format, "application/octet-stream") - headers["Content-Disposition"] = "attachment; filename=%s.%s; filename*=UTF-8''%s.%s" % ( - quote(file_name), book_format, quote(file_name), book_format) + headers["Content-Disposition"] = ('attachment; filename="{}.{}"; filename*=UTF-8\'\'{}.{}').format( + file_name, book_format, file_name, book_format) return do_download_file(book, book_format, client, data1, headers) else: log.error("Book id {} not found for downloading".format(book_id)) diff --git a/cps/isoLanguages.py b/cps/isoLanguages.py index 57473658..a34cd34a 100644 --- a/cps/isoLanguages.py +++ b/cps/isoLanguages.py @@ -15,24 +15,17 @@ # # You should have received a copy of the GNU General Public License # along with this program. If not, see . +import sys from .iso_language_names import LANGUAGE_NAMES as _LANGUAGE_NAMES from . import logger +from .string_helper import strip_whitespaces log = logger.create() try: - from iso639 import languages, __version__ - get = languages.get -except ImportError: from pycountry import languages as pyc_languages - try: - import pkg_resources - __version__ = pkg_resources.get_distribution('pycountry').version + ' (PyCountry)' - del pkg_resources - except (ImportError, Exception): - __version__ = "? (PyCountry)" def _copy_fields(l): l.part1 = getattr(l, 'alpha_2', None) @@ -46,6 +39,11 @@ except ImportError: return _copy_fields(pyc_languages.get(alpha_2=part1)) if name is not None: return _copy_fields(pyc_languages.get(name=name)) +except ImportError as ex: + if sys.version_info >= (3, 12): + print("Python 3.12 isn't compatible with iso-639. Please install pycountry.") + from iso639 import languages + get = languages.get def get_language_names(locale): @@ -69,20 +67,20 @@ def get_language_name(locale, lang_code): return name -def get_language_codes(locale, language_names, remainder=None): - language_names = set(x.strip().lower() for x in language_names if x) +def get_language_code_from_name(locale, language_names, remainder=None): + language_names = set(strip_whitespaces(x).lower() for x in language_names if x) lang = list() - for k, v in get_language_names(locale).items(): - v = v.lower() - if v in language_names: - lang.append(k) - language_names.remove(v) + for key, val in get_language_names(locale).items(): + val = val.lower() + if val in language_names: + lang.append(key) + language_names.remove(val) if remainder is not None and language_names: remainder.extend(language_names) return lang -def get_valid_language_codes(locale, language_names, remainder=None): +def get_valid_language_codes_from_code(locale, language_names, remainder=None): lang = list() if "" in language_names: language_names.remove("") @@ -103,6 +101,6 @@ def get_lang3(lang): ret_value = lang else: ret_value = "" - except KeyError: + except (KeyError, AttributeError): ret_value = lang return ret_value diff --git a/cps/iso_language_names.py b/cps/iso_language_names.py index 4b9a8ef9..9248443e 100644 --- a/cps/iso_language_names.py +++ b/cps/iso_language_names.py @@ -8138,6 +8138,384 @@ LANGUAGE_NAMES = { "zul": "Zulu", "zun": "Zuni" }, + "sl": { + "abk": "abhazijski", + "ace": "achinese", + "ach": "Acoli", + "ada": "Adangme", + "ady": "Adyghe", + "aar": "afarski", + "afh": "Afrihili", + "afr": "afrikanski", + "ain": "Ainu (Japan)", + "aka": "Akan", + "akk": "akadski", + "sqi": "albanščina", + "ale": "aleutski", + "amh": "amharski", + "anp": "Angika", + "ara": "arabski", + "arg": "aragonski", + "arp": "Arapaho", + "arw": "araukanski", + "hye": "armenščina", + "asm": "asamski", + "ast": "Asturian", + "ava": "avarski", + "ave": "avestijski jeziki", + "awa": "Awadhi", + "aym": "Aymara", + "aze": "azerbajdžanski", + "ban": "balijščina", + "bal": "belučijski", + "bam": "bambarski", + "bas": "Basa (Cameroon)", + "bak": "baškirski", + "eus": "baskovščina", + "bej": "Beja", + "bel": "beloruščina", + "bem": "Bemba (Zambia)", + "ben": "bengalščina", + "bit": "Berinomo", + "bho": "Bhojpuri", + "bik": "bikolščina", + "byn": "Bilin", + "bin": "Bini", + "bis": "bislama", + "zbl": "Blissymbols", + "bos": "bošnjaščina", + "bra": "Braj", + "bre": "bretonščina", + "bug": "buginščina", + "bul": "bolgarščina", + "bua": "burjatščina", + "mya": "burmanščina", + "cad": "kadajščina?", + "cat": "katalonščina", + "ceb": "cebuanščina", + "chg": "Chagatai", + "cha": "čamorščina", + "che": "čečenščina", + "chr": "čerokeščina", + "chy": "čejenščina", + "chb": "čibčevščina", + "zho": "kitajščina", + "chn": "Chinook jargon", + "chp": "čipevščina", + "cho": "Choctaw", + "cht": "Cholón", + "chk": "Chuukese", + "chv": "čuvaščina", + "cop": "koptščina", + "cor": "kornijščina", + "cos": "korzijščina", + "cre": "krijščina", + "mus": "Creek", + "hrv": "hrvaščina", + "ces": "češčina", + "dak": "Dakota", + "dan": "danski", + "dar": "Dargwa", + "del": "Delaware", + "div": "Dhivehi", + "din": "Dinka", + "doi": "Dogri (macrolanguage)", + "dgr": "Dogrib", + "dua": "Duala", + "nld": "nizozemščina", + "dse": "Dutch Sign Language", + "dyu": "Dyula", + "dzo": "dzongkha", + "efi": "Efik", + "egy": "egipčanski", + "eka": "Ekajuk", + "elx": "elamščina", + "eng": "angleščina", + "enu": "Enu", + "myv": "Erzya", + "epo": "esperanto", + "est": "estonščina", + "ewe": "evenščina", + "ewo": "Ewondo", + "fan": "Fang (Equatorial Guinea)", + "fat": "Fanti", + "fao": "ferščina", + "fij": "fidžijščina", + "fil": "Filipino", + "fin": "finščina", + "fon": "Fon", + "fra": "francoščina", + "fur": "furlanščina", + "ful": "fulščina", + "gaa": "Ga", + "glg": "Galician", + "lug": "Ganda", + "gay": "gajščina?", + "gba": "Gbaya (Central African Republic)", + "hmj": "Ge", + "gez": "etiopščina?", + "kat": "gruzinščina", + "deu": "nemški", + "gil": "gilbertščina", + "gon": "Gondi", + "gor": "Gorontalo", + "got": "gotščina", + "grb": "Grebo", + "grn": "gvaranijščina", + "guj": "gudžaratščina", + "gwi": "Gwichʼin", + "hai": "haidščina", + "hau": "havščina", + "haw": "havajščina", + "heb": "hebrejščina", + "her": "Herero", + "hil": "hilingajnonščina", + "hin": "hindijščina", + "hmo": "hiri motu", + "hit": "hetitščina", + "hmn": "hmonščina; miaojščina", + "hun": "madžarščina", + "hup": "hupščina", + "iba": "ibanščina", + "isl": "islandščina", + "ido": "Ido", + "ibo": "Igbo", + "ilo": "Iloko", + "ind": "indonezijščina", + "inh": "inguščina", + "ina": "interlingva", + "ile": "Interlingue", + "iku": "inuktituščina", + "ipk": "Inupiaq", + "gle": "irščina", + "ita": "italijanščina", + "jpn": "japonščina", + "jav": "javanščina", + "jrb": "Judeo-Arabic", + "jpr": "Judeo-Persian", + "kbd": "kabardinščina", + "kab": "Kabyle", + "kac": "Kachin", + "kal": "Kalaallisut", + "xal": "Kalmyk", + "kam": "Kamba (Kenya)", + "kan": "kanareščina", + "kau": "Kanuri", + "kaa": "Kara-Kalpak", + "krc": "Karachay-Balkar", + "krl": "Karelian", + "kas": "kašmirščina", + "csb": "Kashubian", + "kaw": "kavi", + "kaz": "kazaščina", + "kha": "Khasi", + "kho": "Khotanese", + "kik": "kikujščina", + "kmb": "Kimbundu", + "kin": "Kinyarwanda", + "kir": "kirgiščina", + "tlh": "Klingon", + "kom": "komijščina", + "kon": "Kongo", + "kok": "Konkani (macrolanguage)", + "kor": "korejščina", + "kos": "Kosraean", + "kpe": "Kpelle", + "kua": "Kuanyama", + "kum": "kumiščina", + "kur": "kurdščina", + "kru": "Kurukh", + "kut": "kutenajščina", + "lad": "ladinščina", + "lah": "Lahnda", + "lam": "Lamba", + "lao": "laoščina", + "lat": "latinščina", + "lav": "latvijščina", + "lez": "lezginščina", + "lim": "Limburgan", + "lin": "lingala", + "lit": "litvanščina", + "jbo": "Lojban", + "loz": "Lozi", + "lub": "Luba-Katanga", + "lua": "lubalulujščina", + "lui": "Luiseno", + "smj": "Lule Sami", + "lun": "Lunda", + "luo": "Luo (Kenya and Tanzania)", + "lus": "Lushai", + "ltz": "Luxembourgish", + "mkd": "makedonščina", + "mad": "madurščina", + "mag": "Magahi", + "mai": "Maithili", + "mak": "makasarščina", + "mlg": "malgaščina", + "msa": "Malay (macrolanguage)", + "mal": "malajalščina", + "mlt": "malteščina", + "mnc": "Manchu", + "mdr": "Mandar", + "man": "Mandingo", + "mni": "manipurščina", + "glv": "manska gelščina", + "mri": "maorščina", + "arn": "Mapudungun", + "mar": "maratščina", + "chm": "Mari (Russia)", + "mah": "Marshallese", + "mwr": "Marwari", + "mas": "masajščina", + "men": "Mende (Sierra Leone)", + "mic": "Mi'kmaq", + "min": "Minangkabau", + "mwl": "Mirandese", + "moh": "mohoščina", + "mdf": "Moksha", + "lol": "Mongo", + "mon": "mongolščina", + "mos": "mosanščina", + "mul": "Več jezikov", + "nqo": "N'Ko", + "nau": "Nauru", + "nav": "navaščina", + "ndo": "Ndonga", + "nap": "napolitanščina", + "nia": "niaščina", + "niu": "niuejščina", + "zxx": "No linguistic content", + "nog": "Nogai", + "nor": "norveščina", + "nob": "Norwegian Bokmål", + "nno": "norveščina; nynorsk", + "nym": "Nyamwezi", + "nya": "Nyanja", + "nyn": "Nyankole", + "nyo": "Nyoro", + "nzi": "Nzima", + "oci": "Occitan (post 1500)", + "oji": "Ojibwa", + "orm": "Oromo", + "osa": "Osage", + "oss": "Ossetian", + "pal": "Pahlavi", + "pau": "palavanščina", + "pli": "Pali", + "pam": "Pampanga", + "pag": "pangasinanščina", + "pan": "Panjabi", + "pap": "papiamentu", + "fas": "perzijščina", + "phn": "feničanščina", + "pon": "Pohnpeian", + "pol": "poljščina", + "por": "portugalđščina", + "pus": "paštu", + "que": "Quechua", + "raj": "radžastanščina", + "rap": "rapanujščina", + "ron": "romunščina", + "roh": "Romansh", + "rom": "romščina", + "run": "rundščina", + "rus": "ruščina", + "smo": "samoanščina", + "sad": "Sandawe", + "sag": "Sango", + "san": "sanskrt", + "sat": "santalščina", + "srd": "sardinščina", + "sas": "Sasak", + "sco": "škotščina", + "sel": "selkupščina", + "srp": "srbščina", + "srr": "Serer", + "shn": "šanščina", + "sna": "šonščina", + "scn": "sicilijanščina", + "sid": "Sidamo", + "bla": "Siksika", + "snd": "sindščina", + "sin": "Sinhala", + "den": "Slave (Athapascan)", + "slk": "slovaščina", + "slv": "slovenščina", + "sog": "Sogdian", + "som": "Somali", + "snk": "Soninke", + "spa": "španščina", + "srn": "Sranan Tongo", + "suk": "Sukuma", + "sux": "sumerščina", + "sun": "sundščina", + "sus": "susuamijščina?", + "swa": "Swahili (macrolanguage)", + "ssw": "svazijščina?", + "swe": "švedščina", + "syr": "sirščina", + "tgl": "tagaloščina", + "tah": "tahitijščina", + "tgk": "tadžiščina", + "tmh": "Tamashek", + "tam": "tamilščina", + "tat": "tatarščina", + "tel": "Telugu", + "ter": "Tereno", + "tet": "Tetum", + "tha": "tajščina", + "bod": "tibetanščina", + "tig": "Tigre", + "tir": "Tigrinya", + "tem": "Timne", + "tiv": "Tiv", + "tli": "Tlingit", + "tpi": "tok pisin", + "tkl": "Tokelau", + "tog": "Tonga (Nyasa)", + "ton": "tonganščina", + "tsi": "tsimšijščina", + "tso": "Tsonga", + "tsn": "Tswana", + "tum": "Tumbuka", + "tur": "turščina", + "tuk": "turkmenščina", + "tvl": "tuvalujščina", + "tyv": "Tuvinian", + "twi": "Twi", + "udm": "Udmurt", + "uga": "ugaritščina", + "uig": "ujgurščina", + "ukr": "ukrajinščina", + "umb": "Umbundu", + "mis": "Uncoded languages", + "und": "nedoločen", + "urd": "urdujščina", + "uzb": "uzbeščina", + "vai": "vajščina", + "ven": "Venda", + "vie": "vietnamščina", + "vol": "Volapük", + "vot": "votjaščina", + "wln": "valonščina", + "war": "Waray (Philippines)", + "was": "Washo", + "cym": "valižanščina", + "wal": "Wolaytta", + "wol": "Wolof", + "xho": "koščina", + "sah": "jakutščina", + "yao": "jaojščina", + "yap": "Yapese", + "yid": "jidiš", + "yor": "jorubščina", + "zap": "Zapotec", + "zza": "Zaza", + "zen": "Zenaga", + "zha": "Zhuang", + "zul": "zulujščina", + "zun": "Zuni" + }, "sv": { "aar": "Afar", "abk": "Abchaziska", diff --git a/cps/jinjia.py b/cps/jinjia.py index f0b3489d..cb51b2bc 100644 --- a/cps/jinjia.py +++ b/cps/jinjia.py @@ -27,7 +27,7 @@ import datetime import mimetypes from uuid import uuid4 -from flask import Blueprint, request, url_for +from flask import Blueprint, request, url_for, g from flask_babel import format_date from .cw_login import current_user @@ -43,6 +43,8 @@ def url_for_other_page(page): args = request.view_args.copy() args['page'] = page for get, val in request.args.items(): + if get == "page": + continue args[get] = val return url_for(request.endpoint, **args) @@ -111,21 +113,12 @@ def yesno(value, yes, no): @jinjia.app_template_filter('formatfloat') def formatfloat(value, decimals=1): - value = 0 if not value else value - return ('{0:.' + str(decimals) + 'f}').format(value).rstrip('0').rstrip('.') - - -@jinjia.app_template_filter('formatseriesindex') -def formatseriesindex_filter(series_index): - if series_index: - try: - if int(series_index) - series_index == 0: - return int(series_index) - else: - return series_index - except (ValueError, TypeError): - return series_index - return 0 + if not value or (isinstance(value, str) and not value.is_numeric()): + return value + formated_value = ('{0:.' + str(decimals) + 'f}').format(value) + if formated_value.endswith('.' + "0" * decimals): + formated_value = formated_value.rstrip('0').rstrip('.') + return formated_value @jinjia.app_template_filter('escapedlink') @@ -179,3 +172,12 @@ def get_cover_srcset(series): url = url_for('web.get_series_cover', series_id=series.id, resolution=shortname, c=cache_timestamp()) srcset.append(f'{url} {resolution}x') return ', '.join(srcset) + + +@jinjia.app_template_filter('music') +def contains_music(book_formats): + result = False + for format in book_formats: + if format.format.lower() in g.constants.EXTENSIONS_AUDIO: + result = True + return result diff --git a/cps/kobo.py b/cps/kobo.py index 3e01f410..cfdc60c9 100644 --- a/cps/kobo.py +++ b/cps/kobo.py @@ -18,7 +18,7 @@ # along with this program. If not, see . import base64 -import datetime +from datetime import datetime, timezone import os import uuid import zipfile @@ -47,7 +47,7 @@ import requests from . import config, logger, kobo_auth, db, calibre_db, helper, shelf as shelf_lib, ub, csrf, kobo_sync_status from . import isoLanguages from .epub import get_epub_layout -from .constants import COVER_THUMBNAIL_SMALL +from .constants import COVER_THUMBNAIL_SMALL, COVER_THUMBNAIL_MEDIUM, COVER_THUMBNAIL_LARGE from .helper import get_download_link from .services import SyncToken as SyncToken from .web import download_required @@ -106,24 +106,29 @@ def make_request_to_kobo_store(sync_token=None): return store_response -def redirect_or_proxy_request(): +def redirect_or_proxy_request(auth=False): if config.config_kobo_proxy: - if request.method == "GET": - return redirect(get_store_url_for_current_request(), 307) - else: - # The Kobo device turns other request types into GET requests on redirects, - # so we instead proxy to the Kobo store ourselves. - store_response = make_request_to_kobo_store() + try: + if request.method == "GET": + alfa = redirect(get_store_url_for_current_request(), 307) + return alfa + else: + # The Kobo device turns other request types into GET requests on redirects, + # so we instead proxy to the Kobo store ourselves. + store_response = make_request_to_kobo_store() - response_headers = store_response.headers - for header_key in CONNECTION_SPECIFIC_HEADERS: - response_headers.pop(header_key, default=None) + response_headers = store_response.headers + for header_key in CONNECTION_SPECIFIC_HEADERS: + response_headers.pop(header_key, default=None) - return make_response( - store_response.content, store_response.status_code, response_headers.items() - ) - else: - return make_response(jsonify({})) + return make_response( + store_response.content, store_response.status_code, response_headers.items() + ) + except Exception as e: + log.error("Failed to receive or parse response from Kobo's endpoint: {}".format(e)) + if auth: + return make_calibre_web_auth_response() + return make_response(jsonify({})) def convert_to_kobo_timestamp_string(timestamp): @@ -131,7 +136,7 @@ def convert_to_kobo_timestamp_string(timestamp): return timestamp.strftime("%Y-%m-%dT%H:%M:%SZ") except AttributeError as exc: log.debug("Timestamp not valid: {}".format(exc)) - return datetime.datetime.utcnow().strftime("%Y-%m-%dT%H:%M:%SZ") + return datetime.now(timezone.utc).strftime("%Y-%m-%dT%H:%M:%SZ") @kobo.route("/v1/library/sync") @@ -150,15 +155,15 @@ def HandleSyncRequest(): # if no books synced don't respect sync_token if not ub.session.query(ub.KoboSyncedBooks).filter(ub.KoboSyncedBooks.user_id == current_user.id).count(): - sync_token.books_last_modified = datetime.datetime.min - sync_token.books_last_created = datetime.datetime.min - sync_token.reading_state_last_modified = datetime.datetime.min + sync_token.books_last_modified = datetime.min + sync_token.books_last_created = datetime.min + sync_token.reading_state_last_modified = datetime.min new_books_last_modified = sync_token.books_last_modified # needed for sync selected shelfs only new_books_last_created = sync_token.books_last_created # needed to distinguish between new and changed entitlement new_reading_state_last_modified = sync_token.reading_state_last_modified - new_archived_last_modified = datetime.datetime.min + new_archived_last_modified = datetime.min sync_results = [] # We reload the book database so that the user gets a fresh view of the library @@ -323,7 +328,7 @@ def generate_sync_response(sync_token, sync_results, set_cont=False): sync_token.to_headers(extra_headers) # log.debug("Kobo Sync Content: {}".format(sync_results)) - # jsonify decodes the unicode string different to what kobo expects + # jsonify decodes the Unicode string different to what kobo expects response = make_response(json.dumps(sync_results), extra_headers) response.headers["Content-Type"] = "application/json; charset=utf-8" return response @@ -375,7 +380,7 @@ def create_book_entitlement(book, archived): book_uuid = str(book.uuid) return { "Accessibility": "Full", - "ActivePeriod": {"From": convert_to_kobo_timestamp_string(datetime.datetime.utcnow())}, + "ActivePeriod": {"From": convert_to_kobo_timestamp_string(datetime.now(timezone.utc))}, "Created": convert_to_kobo_timestamp_string(book.timestamp), "CrossRevisionId": book_uuid, "Id": book_uuid, @@ -423,7 +428,7 @@ def get_series(book): def get_seriesindex(book): - return book.series_index or 1 + return book.series_index if isinstance(book.series_index, float) else 1 def get_language(book): @@ -486,14 +491,16 @@ def get_metadata(book): if get_series(book): name = get_series(book) - metadata["Series"] = { - "Name": get_series(book), - "Number": get_seriesindex(book), # ToDo Check int() ? - "NumberFloat": float(get_seriesindex(book)), - # Get a deterministic id based on the series name. - "Id": str(uuid.uuid3(uuid.NAMESPACE_DNS, name)), - } - + try: + metadata["Series"] = { + "Name": get_series(book), + "Number": get_seriesindex(book), # ToDo Check int() ? + "NumberFloat": float(get_seriesindex(book)), + # Get a deterministic id based on the series name. + "Id": str(uuid.uuid3(uuid.NAMESPACE_DNS, name)), + } + except Exception as e: + print(e) return metadata @@ -725,7 +732,7 @@ def sync_shelves(sync_token, sync_results, only_kobo_shelves=False): ub.session_commit() -# Creates a Kobo "Tag" object from a ub.Shelf object +# Creates a Kobo "Tag" object from an ub.Shelf object def create_kobo_tag(shelf): tag = { "Created": convert_to_kobo_timestamp_string(shelf.created), @@ -795,7 +802,7 @@ def HandleStateRequest(book_uuid): if new_book_read_status == ub.ReadBook.STATUS_IN_PROGRESS \ and new_book_read_status != book_read.read_status: book_read.times_started_reading += 1 - book_read.last_time_started_reading = datetime.datetime.utcnow() + book_read.last_time_started_reading = datetime.now(timezone.utc) book_read.read_status = new_book_read_status update_results_response["StatusInfoResult"] = {"Result": "Success"} except (KeyError, TypeError, ValueError, StatementError): @@ -903,7 +910,12 @@ def get_current_bookmark_response(current_bookmark): @requires_kobo_auth def HandleCoverImageRequest(book_uuid, width, height, Quality, isGreyscale): try: - resolution = None if int(height) > 1000 else COVER_THUMBNAIL_SMALL + if int(height) > 1000: + resolution = COVER_THUMBNAIL_LARGE + elif int(height) > 500: + resolution = COVER_THUMBNAIL_MEDIUM + else: + resolution = COVER_THUMBNAIL_SMALL except ValueError: log.error("Requested height %s of book %s is invalid" % (book_uuid, height)) resolution = COVER_THUMBNAIL_SMALL @@ -948,7 +960,7 @@ def HandleBookDeletionRequest(book_uuid): # TODO: Implement the following routes @csrf.exempt -@kobo.route("/v1/library/", methods=["DELETE", "GET"]) +@kobo.route("/v1/library/", methods=["DELETE", "GET", "POST"]) def HandleUnimplementedRequest(dummy=None): log.debug("Unimplemented Library Request received: %s (request is forwarded to kobo if configured)", request.base_url) @@ -1035,7 +1047,7 @@ def HandleAuthRequest(): log.debug('Kobo Auth request') if config.config_kobo_proxy: try: - return redirect_or_proxy_request() + return redirect_or_proxy_request(auth=True) except Exception: log.error("Failed to receive or parse response from Kobo's auth endpoint. Falling back to un-proxied mode.") return make_calibre_web_auth_response() @@ -1119,25 +1131,37 @@ def download_book(book_id, book_format): def NATIVE_KOBO_RESOURCES(): return { - "account_page": "https://secure.kobobooks.com/profile", + "account_page": "https://www.kobo.com/account/settings", "account_page_rakuten": "https://my.rakuten.co.jp/", + "add_device": "https://storeapi.kobo.com/v1/user/add-device", "add_entitlement": "https://storeapi.kobo.com/v1/library/{RevisionIds}", "affiliaterequest": "https://storeapi.kobo.com/v1/affiliate", + "assets": "https://storeapi.kobo.com/v1/assets", + "audiobook": "https://storeapi.kobo.com/v1/products/audiobooks/{ProductId}", + "audiobook_detail_page": "https://www.kobo.com/{region}/{language}/audiobook/{slug}", + "audiobook_landing_page": "https://www.kobo.com/{region}/{language}/audiobooks", + "audiobook_preview": "https://storeapi.kobo.com/v1/products/audiobooks/{Id}/preview", + "audiobook_purchase_withcredit": "https://storeapi.kobo.com/v1/store/audiobook/{Id}", "audiobook_subscription_orange_deal_inclusion_url": "https://authorize.kobo.com/inclusion", "authorproduct_recommendations": "https://storeapi.kobo.com/v1/products/books/authors/recommendations", "autocomplete": "https://storeapi.kobo.com/v1/products/autocomplete", - "blackstone_header": {"key": "x-amz-request-payer", "value": "requester"}, + "blackstone_header": { + "key": "x-amz-request-payer", + "value": "requester" + }, "book": "https://storeapi.kobo.com/v1/products/books/{ProductId}", - "book_detail_page": "https://store.kobobooks.com/{culture}/ebook/{slug}", - "book_detail_page_rakuten": "https://books.rakuten.co.jp/rk/{crossrevisionid}", - "book_landing_page": "https://store.kobobooks.com/ebooks", + "book_detail_page": "https://www.kobo.com/{region}/{language}/ebook/{slug}", + "book_detail_page_rakuten": "http://books.rakuten.co.jp/rk/{crossrevisionid}", + "book_landing_page": "https://www.kobo.com/ebooks", "book_subscription": "https://storeapi.kobo.com/v1/products/books/subscriptions", + "browse_history": "https://storeapi.kobo.com/v1/user/browsehistory", "categories": "https://storeapi.kobo.com/v1/categories", - "categories_page": "https://store.kobobooks.com/ebooks/categories", + "categories_page": "https://www.kobo.com/ebooks/categories", "category": "https://storeapi.kobo.com/v1/categories/{CategoryId}", "category_featured_lists": "https://storeapi.kobo.com/v1/categories/{CategoryId}/featured", "category_products": "https://storeapi.kobo.com/v1/categories/{CategoryId}/products", "checkout_borrowed_book": "https://storeapi.kobo.com/v1/library/borrow", + "client_authd_referral": "https://authorize.kobo.com/api/AuthenticatedReferral/client/v1/getLink", "configuration_data": "https://storeapi.kobo.com/v1/configuration", "content_access_book": "https://storeapi.kobo.com/v1/products/books/{ProductId}/access", "customer_care_live_chat": "https://v2.zopim.com/widget/livechat.html?key=Y6gwUmnu4OATxN3Tli4Av9bYN319BTdO", @@ -1148,92 +1172,109 @@ def NATIVE_KOBO_RESOURCES(): "delete_tag_items": "https://storeapi.kobo.com/v1/library/tags/{TagId}/items/delete", "device_auth": "https://storeapi.kobo.com/v1/auth/device", "device_refresh": "https://storeapi.kobo.com/v1/auth/refresh", - "dictionary_host": "https://kbdownload1-a.akamaihd.net", + "dictionary_host": "https://ereaderfiles.kobo.com", "discovery_host": "https://discovery.kobobooks.com", + "ereaderdevices": "https://storeapi.kobo.com/v2/products/EReaderDeviceFeeds", "eula_page": "https://www.kobo.com/termsofuse?style=onestore", "exchange_auth": "https://storeapi.kobo.com/v1/auth/exchange", "external_book": "https://storeapi.kobo.com/v1/products/books/external/{Ids}", - "facebook_sso_page": - "https://authorize.kobo.com/signin/provider/Facebook/login?returnUrl=http://store.kobobooks.com/", + "facebook_sso_page": "https://authorize.kobo.com/signin/provider/Facebook/login?returnUrl=http://kobo.com/", "featured_list": "https://storeapi.kobo.com/v1/products/featured/{FeaturedListId}", "featured_lists": "https://storeapi.kobo.com/v1/products/featured", "free_books_page": { "EN": "https://www.kobo.com/{region}/{language}/p/free-ebooks", "FR": "https://www.kobo.com/{region}/{language}/p/livres-gratuits", "IT": "https://www.kobo.com/{region}/{language}/p/libri-gratuiti", - "NL": "https://www.kobo.com/{region}/{language}/" - "List/bekijk-het-overzicht-van-gratis-ebooks/QpkkVWnUw8sxmgjSlCbJRg", - "PT": "https://www.kobo.com/{region}/{language}/p/livros-gratis", + "NL": "https://www.kobo.com/{region}/{language}/List/bekijk-het-overzicht-van-gratis-ebooks/QpkkVWnUw8sxmgjSlCbJRg", + "PT": "https://www.kobo.com/{region}/{language}/p/livros-gratis" }, "fte_feedback": "https://storeapi.kobo.com/v1/products/ftefeedback", + "funnel_metrics": "https://storeapi.kobo.com/v1/funnelmetrics", + "get_download_keys": "https://storeapi.kobo.com/v1/library/downloadkeys", + "get_download_link": "https://storeapi.kobo.com/v1/library/downloadlink", "get_tests_request": "https://storeapi.kobo.com/v1/analytics/gettests", "giftcard_epd_redeem_url": "https://www.kobo.com/{storefront}/{language}/redeem-ereader", "giftcard_redeem_url": "https://www.kobo.com/{storefront}/{language}/redeem", - "help_page": "https://www.kobo.com/help", - "kobo_audiobooks_enabled": "False", + "gpb_flow_enabled": "False", + "help_page": "http://www.kobo.com/help", + "image_host": "//cdn.kobo.com/book-images/", + "image_url_quality_template": "https://cdn.kobo.com/book-images/{ImageId}/{Width}/{Height}/{Quality}/{IsGreyscale}/image.jpg", + "image_url_template": "https://cdn.kobo.com/book-images/{ImageId}/{Width}/{Height}/false/image.jpg", + "kobo_audiobooks_credit_redemption": "False", + "kobo_audiobooks_enabled": "True", "kobo_audiobooks_orange_deal_enabled": "False", "kobo_audiobooks_subscriptions_enabled": "False", - "kobo_nativeborrow_enabled": "True", + "kobo_display_price": "True", + "kobo_dropbox_link_account_enabled": "False", + "kobo_google_tax": "False", + "kobo_googledrive_link_account_enabled": "False", + "kobo_nativeborrow_enabled": "False", + "kobo_onedrive_link_account_enabled": "False", "kobo_onestorelibrary_enabled": "False", + "kobo_privacyCentre_url": "https://www.kobo.com/privacy", "kobo_redeem_enabled": "True", "kobo_shelfie_enabled": "False", - "kobo_subscriptions_enabled": "False", - "kobo_superpoints_enabled": "False", + "kobo_subscriptions_enabled": "True", + "kobo_superpoints_enabled": "True", "kobo_wishlist_enabled": "True", "library_book": "https://storeapi.kobo.com/v1/user/library/books/{LibraryItemId}", "library_items": "https://storeapi.kobo.com/v1/user/library", "library_metadata": "https://storeapi.kobo.com/v1/library/{Ids}/metadata", "library_prices": "https://storeapi.kobo.com/v1/user/library/previews/prices", - "library_stack": "https://storeapi.kobo.com/v1/user/library/stacks/{LibraryItemId}", + "library_search": "https://storeapi.kobo.com/v1/library/search", "library_sync": "https://storeapi.kobo.com/v1/library/sync", - "love_dashboard_page": "https://store.kobobooks.com/{culture}/kobosuperpoints", - "love_points_redemption_page": - "https://store.kobobooks.com/{culture}/KoboSuperPointsRedemption?productId={ProductId}", - "magazine_landing_page": "https://store.kobobooks.com/emagazines", + "love_dashboard_page": "https://www.kobo.com/{region}/{language}/kobosuperpoints", + "love_points_redemption_page": "https://www.kobo.com/{region}/{language}/KoboSuperPointsRedemption?productId={ProductId}", + "magazine_landing_page": "https://www.kobo.com/emagazines", + "more_sign_in_options": "https://authorize.kobo.com/signin?returnUrl=http://kobo.com/#allProviders", + "notebooks": "https://storeapi.kobo.com/api/internal/notebooks", "notifications_registration_issue": "https://storeapi.kobo.com/v1/notifications/registration", "oauth_host": "https://oauth.kobo.com", - "overdrive_account": "https://auth.overdrive.com/account", - "overdrive_library": "https://{libraryKey}.auth.overdrive.com/library", - "overdrive_library_finder_host": "https://libraryfinder.api.overdrive.com", - "overdrive_thunder_host": "https://thunder.api.overdrive.com", - "password_retrieval_page": "https://www.kobobooks.com/passwordretrieval.html", + "password_retrieval_page": "https://www.kobo.com/passwordretrieval.html", + "personalizedrecommendations": "https://storeapi.kobo.com/v2/users/personalizedrecommendations", + "pocket_link_account_start": "https://authorize.kobo.com/{region}/{language}/linkpocket", "post_analytics_event": "https://storeapi.kobo.com/v1/analytics/event", + "ppx_purchasing_url": "https://purchasing.kobo.com", "privacy_page": "https://www.kobo.com/privacypolicy?style=onestore", "product_nextread": "https://storeapi.kobo.com/v1/products/{ProductIds}/nextread", "product_prices": "https://storeapi.kobo.com/v1/products/{ProductIds}/prices", "product_recommendations": "https://storeapi.kobo.com/v1/products/{ProductId}/recommendations", "product_reviews": "https://storeapi.kobo.com/v1/products/{ProductIds}/reviews", "products": "https://storeapi.kobo.com/v1/products", - "provider_external_sign_in_page": - "https://authorize.kobo.com/ExternalSignIn/{providerName}?returnUrl=http://store.kobobooks.com/", - "purchase_buy": "https://www.kobo.com/checkout/createpurchase/", - "purchase_buy_templated": "https://www.kobo.com/{culture}/checkout/createpurchase/{ProductId}", + "productsv2": "https://storeapi.kobo.com/v2/products", + "provider_external_sign_in_page": "https://authorize.kobo.com/ExternalSignIn/{providerName}?returnUrl=http://kobo.com/", "quickbuy_checkout": "https://storeapi.kobo.com/v1/store/quickbuy/{PurchaseId}/checkout", "quickbuy_create": "https://storeapi.kobo.com/v1/store/quickbuy/purchase", + "rakuten_token_exchange": "https://storeapi.kobo.com/v1/auth/rakuten_token_exchange", "rating": "https://storeapi.kobo.com/v1/products/{ProductId}/rating/{Rating}", + "reading_services_host": "https://readingservices.kobo.com", "reading_state": "https://storeapi.kobo.com/v1/library/{Ids}/state", - "redeem_interstitial_page": "https://store.kobobooks.com", - "registration_page": "https://authorize.kobo.com/signup?returnUrl=http://store.kobobooks.com/", + "redeem_interstitial_page": "https://www.kobo.com", + "registration_page": "https://authorize.kobo.com/signup?returnUrl=http://kobo.com/", "related_items": "https://storeapi.kobo.com/v1/products/{Id}/related", "remaining_book_series": "https://storeapi.kobo.com/v1/products/books/series/{SeriesId}", "rename_tag": "https://storeapi.kobo.com/v1/library/tags/{TagId}", "review": "https://storeapi.kobo.com/v1/products/reviews/{ReviewId}", "review_sentiment": "https://storeapi.kobo.com/v1/products/reviews/{ReviewId}/sentiment/{Sentiment}", "shelfie_recommendations": "https://storeapi.kobo.com/v1/user/recommendations/shelfie", - "sign_in_page": "https://authorize.kobo.com/signin?returnUrl=http://store.kobobooks.com/", + "sign_in_page": "https://authorize.kobo.com/signin?returnUrl=http://kobo.com/", "social_authorization_host": "https://social.kobobooks.com:8443", "social_host": "https://social.kobobooks.com", - "stacks_host_productId": "https://store.kobobooks.com/collections/byproductid/", "store_home": "www.kobo.com/{region}/{language}", - "store_host": "store.kobobooks.com", - "store_newreleases": "https://store.kobobooks.com/{culture}/List/new-releases/961XUjtsU0qxkFItWOutGA", - "store_search": "https://store.kobobooks.com/{culture}/Search?Query={query}", - "store_top50": "https://store.kobobooks.com/{culture}/ebooks/Top", + "store_host": "www.kobo.com", + "store_newreleases": "https://www.kobo.com/{region}/{language}/List/new-releases/961XUjtsU0qxkFItWOutGA", + "store_search": "https://www.kobo.com/{region}/{language}/Search?Query={query}", + "store_top50": "https://www.kobo.com/{region}/{language}/ebooks/Top", + "subs_landing_page": "https://www.kobo.com/{region}/{language}/plus", + "subs_management_page": "https://www.kobo.com/{region}/{language}/account/subscriptions", + "subs_plans_page": "https://www.kobo.com/{region}/{language}/plus/plans", + "subs_purchase_buy_templated": "https://www.kobo.com/{region}/{language}/Checkoutoption/{ProductId}/{TierId}", "tag_items": "https://storeapi.kobo.com/v1/library/tags/{TagId}/Items", "tags": "https://storeapi.kobo.com/v1/library/tags", "taste_profile": "https://storeapi.kobo.com/v1/products/tasteprofile", + "terms_of_sale_page": "https://authorize.kobo.com/{region}/{language}/terms/termsofsale", "update_accessibility_to_preview": "https://storeapi.kobo.com/v1/library/{EntitlementIds}/preview", - "use_one_store": "False", + "use_one_store": "True", "user_loyalty_benefits": "https://storeapi.kobo.com/v1/user/loyalty/benefits", "user_platform": "https://storeapi.kobo.com/v1/user/platform", "user_profile": "https://storeapi.kobo.com/v1/user/profile", @@ -1241,6 +1282,6 @@ def NATIVE_KOBO_RESOURCES(): "user_recommendations": "https://storeapi.kobo.com/v1/user/recommendations", "user_reviews": "https://storeapi.kobo.com/v1/user/reviews", "user_wishlist": "https://storeapi.kobo.com/v1/user/wishlist", - "userguide_host": "https://kbdownload1-a.akamaihd.net", - "wishlist_page": "https://store.kobobooks.com/{region}/{language}/account/wishlist", + "userguide_host": "https://ereaderfiles.kobo.com", + "wishlist_page": "https://www.kobo.com/{region}/{language}/account/wishlist" } diff --git a/cps/kobo_auth.py b/cps/kobo_auth.py index f99bf77c..53f35dcc 100644 --- a/cps/kobo_auth.py +++ b/cps/kobo_auth.py @@ -22,7 +22,7 @@ This module also includes research notes into the auth protocol used by Kobo devices. Log-in: -When first booting a Kobo device the user must sign into a Kobo (or affiliate) account. +When first booting a Kobo device the user must log in to a Kobo (or affiliate) account. Upon successful sign-in, the user is redirected to https://auth.kobobooks.com/CrossDomainSignIn?id= which serves the following response: @@ -41,7 +41,7 @@ issue for a few years now https://www.mobileread.com/forums/showpost.php?p=34768 will still grant access given the userkey.) Official Kobo Store Api authorization: -* For most of the endpoints we care about (sync, metadata, tags, etc), the userKey is +* For most of the endpoints we care about (sync, metadata, tags, etc.), the userKey is passed in the x-kobo-userkey header, and is sufficient to authorize the API call. * Some endpoints (e.g: AnnotationService) instead make use of Bearer tokens pass through an authorization header. To get a BearerToken, the device makes a POST request to the diff --git a/cps/kobo_sync_status.py b/cps/kobo_sync_status.py index ef732aaa..357b84ef 100644 --- a/cps/kobo_sync_status.py +++ b/cps/kobo_sync_status.py @@ -19,7 +19,7 @@ from .cw_login import current_user from . import ub -import datetime +from datetime import datetime, timezone from sqlalchemy.sql.expression import or_, and_, true # from sqlalchemy import exc @@ -58,7 +58,7 @@ def change_archived_books(book_id, state=None, message=None): archived_book = ub.ArchivedBook(user_id=current_user.id, book_id=book_id) archived_book.is_archived = state if state else not archived_book.is_archived - archived_book.last_modified = datetime.datetime.utcnow() # toDo. Check utc timestamp + archived_book.last_modified = datetime.now(timezone.utc) # toDo. Check utc timestamp ub.session.merge(archived_book) ub.session_commit(message) diff --git a/cps/logger.py b/cps/logger.py index 9aa076d4..7ce2dbc2 100644 --- a/cps/logger.py +++ b/cps/logger.py @@ -29,7 +29,7 @@ from .constants import CONFIG_DIR as _CONFIG_DIR ACCESS_FORMATTER_GEVENT = Formatter("%(message)s") ACCESS_FORMATTER_TORNADO = Formatter("[%(asctime)s] %(message)s") -FORMATTER = Formatter("[%(asctime)s] %(levelname)5s {%(name)s:%(lineno)d} %(message)s") +FORMATTER = Formatter("[%(asctime)s] %(levelname)5s {%(filename)s:%(lineno)d} %(message)s") DEFAULT_LOG_LEVEL = logging.INFO DEFAULT_LOG_FILE = os.path.join(_CONFIG_DIR, "calibre-web.log") DEFAULT_ACCESS_LOG = os.path.join(_CONFIG_DIR, "access.log") @@ -42,18 +42,12 @@ logging.addLevelName(logging.CRITICAL, "CRIT") class _Logger(logging.Logger): - def error_or_exception(self, message, stacklevel=2, *args, **kwargs): + def error_or_exception(self, message, stacklevel=1, *args, **kwargs): is_debug = self.getEffectiveLevel() <= logging.DEBUG - if sys.version_info > (3, 7): - if is_debug: - self.exception(message, stacklevel=stacklevel, *args, **kwargs) - else: - self.error(message, stacklevel=stacklevel, *args, **kwargs) + if not is_debug: + self.exception(message, stacklevel=stacklevel, *args, **kwargs) else: - if is_debug: - self.exception(message, stack_info=True, *args, **kwargs) - else: - self.error(message, *args, **kwargs) + self.error(message, stacklevel=stacklevel, *args, **kwargs) def debug_no_auth(self, message, *args, **kwargs): message = message.strip("\r\n") diff --git a/cps/main.py b/cps/main.py index 1b3b40b4..44e563f9 100644 --- a/cps/main.py +++ b/cps/main.py @@ -31,6 +31,7 @@ def main(): app = create_app() from .web import web + from .basic import basic from .opds import opds from .admin import admi from .gdrive import gdrive @@ -64,6 +65,7 @@ def main(): app.register_blueprint(search) app.register_blueprint(tasks) app.register_blueprint(web) + app.register_blueprint(basic) app.register_blueprint(opds) limiter.limit("3/minute", key_func=request_username)(opds) app.register_blueprint(jinjia) diff --git a/cps/metadata_provider/amazon.py b/cps/metadata_provider/amazon.py index 843a9d76..5b07819c 100644 --- a/cps/metadata_provider/amazon.py +++ b/cps/metadata_provider/amazon.py @@ -38,14 +38,16 @@ class Amazon(Metadata): __name__ = "Amazon" __id__ = "amazon" headers = {'upgrade-insecure-requests': '1', - 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/97.0.4692.71 Safari/537.36', - 'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9', - 'sec-gpc': '1', - 'sec-fetch-site': 'none', - 'sec-fetch-mode': 'navigate', - 'sec-fetch-user': '?1', - 'sec-fetch-dest': 'document', - 'accept-encoding': 'gzip, deflate, br', + 'user-agent': 'Mozilla/5.0 (X11; Linux x86_64; rv:130.0) Gecko/20100101 Firefox/130.0', + 'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/png,image/svg+xml,*/*;q=0.8', + 'Sec-Fetch-Site': 'same-origin', + 'Sec-Fetch-Mode': 'navigate', + 'Sec-Fetch-User': '?1', + 'Sec-Fetch-Dest': 'document', + 'Upgrade-Insecure-Requests': '1', + 'Alt-Used' : 'www.amazon.com', + 'Priority' : 'u=0, i', + 'accept-encoding': 'gzip, deflate, br, zstd', 'accept-language': 'en-US,en;q=0.9'} session = requests.Session() session.headers=headers @@ -53,7 +55,6 @@ class Amazon(Metadata): def search( self, query: str, generic_cover: str = "", locale: str = "en" ) -> Optional[List[MetaRecord]]: - #timer=time() def inner(link, index) -> [dict, int]: with self.session as session: try: @@ -61,11 +62,11 @@ class Amazon(Metadata): r.raise_for_status() except Exception as ex: log.warning(ex) - return None + return [] long_soup = BS(r.text, "lxml") #~4sec :/ - soup2 = long_soup.find("div", attrs={"cel_widget_id": "dpx-books-ppd_csm_instrumentation_wrapper"}) + soup2 = long_soup.find("div", attrs={"cel_widget_id": "dpx-ppd_csm_instrumentation_wrapper"}) if soup2 is None: - return None + return [] try: match = MetaRecord( title = "", @@ -88,7 +89,7 @@ class Amazon(Metadata): soup2.find("div", attrs={"data-feature-name": "bookDescription"}).stripped_strings)\ .replace("\xa0"," ")[:-9].strip().strip("\n") except (AttributeError, TypeError): - return None # if there is no description it is not a book and therefore should be ignored + return [] # if there is no description it is not a book and therefore should be ignored try: match.title = soup2.find("span", attrs={"id": "productTitle"}).text except (AttributeError, TypeError): @@ -107,13 +108,13 @@ class Amazon(Metadata): except (AttributeError, ValueError): match.rating = 0 try: - match.cover = soup2.find("img", attrs={"class": "a-dynamic-image frontImage"})["src"] + match.cover = soup2.find("img", attrs={"class": "a-dynamic-image"})["src"] except (AttributeError, TypeError): match.cover = "" return match, index except Exception as e: log.error_or_exception(e) - return None + return [] val = list() if self.active: @@ -133,7 +134,7 @@ class Amazon(Metadata): links_list = [next(filter(lambda i: "digital-text" in i["href"], x.findAll("a")))["href"] for x in soup.findAll("div", attrs={"data-component-type": "s-search-result"})] with concurrent.futures.ThreadPoolExecutor(max_workers=5) as executor: - fut = {executor.submit(inner, link, index) for index, link in enumerate(links_list[:5])} - val = list(map(lambda x : x.result() ,concurrent.futures.as_completed(fut))) + fut = {executor.submit(inner, link, index) for index, link in enumerate(links_list[:3])} + val = list(map(lambda x : x.result(), concurrent.futures.as_completed(fut))) result = list(filter(lambda x: x, val)) return [x[0] for x in sorted(result, key=itemgetter(1))] #sort by amazons listing order for best relevance diff --git a/cps/metadata_provider/google.py b/cps/metadata_provider/google.py index ba7e10af..df9c0dc2 100644 --- a/cps/metadata_provider/google.py +++ b/cps/metadata_provider/google.py @@ -54,7 +54,7 @@ class Google(Metadata): results.raise_for_status() except Exception as e: log.warning(e) - return None + return [] for result in results.json().get("items", []): val.append( self._parse_search_result( diff --git a/cps/oauth.py b/cps/oauth.py index 0caa61ec..22e92733 100644 --- a/cps/oauth.py +++ b/cps/oauth.py @@ -32,7 +32,7 @@ class OAuthBackend(SQLAlchemyBackend): Stores and retrieves OAuth tokens using a relational database through the `SQLAlchemy`_ ORM. - .. _SQLAlchemy: https://www.sqlalchemy.org/ + _SQLAlchemy: https://www.sqlalchemy.org/ """ def __init__(self, model, session, provider_id, user=None, user_id=None, user_required=None, anon_user=None, diff --git a/cps/opds.py b/cps/opds.py index 3d8c78b9..6dea544a 100644 --- a/cps/opds.py +++ b/cps/opds.py @@ -21,10 +21,10 @@ # along with this program. If not, see . import datetime -import json +# import json from urllib.parse import unquote_plus -from flask import Blueprint, request, render_template, make_response, abort, Response, g +from flask import Blueprint, request, render_template, make_response, abort, g, jsonify from flask_babel import get_locale from flask_babel import gettext as _ @@ -424,11 +424,8 @@ def feed_shelf(book_id): @requires_basic_auth_if_no_ano def opds_download_link(book_id, book_format): if not auth.current_user().role_download(): - return abort(403) - if "Kobo" in request.headers.get('User-Agent'): - client = "kobo" - else: - client = "" + return abort(401) + client = "kobo" if "Kobo" in request.headers.get('User-Agent') else "" return get_download_link(book_id, book_format.lower(), client) @@ -454,7 +451,7 @@ def get_database_stats(): stat['authors'] = calibre_db.session.query(db.Authors).count() stat['categories'] = calibre_db.session.query(db.Tags).count() stat['series'] = calibre_db.session.query(db.Series).count() - return Response(json.dumps(stat), mimetype="application/json") + return make_response(jsonify(stat)) @opds.route("/opds/thumb_240_240/") diff --git a/cps/reverseproxy.py b/cps/reverseproxy.py index 4acb8e45..887590bf 100644 --- a/cps/reverseproxy.py +++ b/cps/reverseproxy.py @@ -41,9 +41,9 @@ class ReverseProxied(object): """Wrap the application in this middleware and configure the front-end server to add these headers, to let you quietly bind this to a URL other than / and to an HTTP scheme that is - different than what is used locally. + different from what is used locally. - Code courtesy of: http://flask.pocoo.org/snippets/35/ + Code courtesy of: https://flask.pocoo.org/snippets/35/ In nginx: location /myprefix { diff --git a/cps/search.py b/cps/search.py index 6054ec9e..29dda16d 100644 --- a/cps/search.py +++ b/cps/search.py @@ -24,9 +24,9 @@ from flask_babel import format_date from flask_babel import gettext as _ from sqlalchemy.sql.expression import func, not_, and_, or_, text, true from sqlalchemy.sql.functions import coalesce -from sqlalchemy import exists from . import logger, db, calibre_db, config, ub +from .string_helper import strip_whitespaces from .usermanagement import login_required_if_no_ano from .render_template import render_title_template from .pagination import Pagination @@ -244,7 +244,8 @@ def render_adv_search_results(term, offset=None, order=None, limit=None): pagination = None cc = calibre_db.get_cc_columns(config, filter_config_custom_read=True) - calibre_db.session.connection().connection.connection.create_function("lower", 1, db.lcase) + calibre_db.create_functions() + # calibre_db.session.connection().connection.connection.create_function("lower", 1, db.lcase) query = calibre_db.generate_linked_query(config.config_read_column, db.Books) q = query.outerjoin(db.books_series_link, db.Books.id == db.books_series_link.c.book)\ .outerjoin(db.Series)\ @@ -257,21 +258,21 @@ def render_adv_search_results(term, offset=None, order=None, limit=None): tags['include_' + element] = term.get('include_' + element) tags['exclude_' + element] = term.get('exclude_' + element) - author_name = term.get("author_name") - book_title = term.get("book_title") + author_name = term.get("authors") + book_title = term.get("title") publisher = term.get("publisher") pub_start = term.get("publishstart") pub_end = term.get("publishend") rating_low = term.get("ratinghigh") rating_high = term.get("ratinglow") - description = term.get("comment") + description = term.get("comments") read_status = term.get("read_status") if author_name: - author_name = author_name.strip().lower().replace(',', '|') + author_name = strip_whitespaces(author_name).lower().replace(',', '|') if book_title: - book_title = book_title.strip().lower() + book_title = strip_whitespaces(book_title).lower() if publisher: - publisher = publisher.strip().lower() + publisher = strip_whitespaces(publisher).lower() search_term = [] cc_present = False diff --git a/cps/search_metadata.py b/cps/search_metadata.py index b818a87c..ae3c9faf 100644 --- a/cps/search_metadata.py +++ b/cps/search_metadata.py @@ -23,7 +23,7 @@ import json import os import sys -from flask import Blueprint, Response, request, url_for +from flask import Blueprint, request, url_for, make_response, jsonify from .cw_login import current_user from flask_babel import get_locale from sqlalchemy.exc import InvalidRequestError, OperationalError @@ -33,7 +33,6 @@ from cps.services.Metadata import Metadata from . import constants, logger, ub, web_server from .usermanagement import user_login_required -# current_milli_time = lambda: int(round(time() * 1000)) meta = Blueprint("metadata", __name__) @@ -90,7 +89,7 @@ def metadata_provider(): provider.append( {"name": c.__name__, "active": ac, "initial": ac, "id": c.__id__} ) - return Response(json.dumps(provider), mimetype="application/json") + return make_response(jsonify(provider)) @meta.route("/metadata/provider", methods=["POST"]) @@ -115,9 +114,7 @@ def metadata_change_active_provider(prov_name): provider = next((c for c in cl if c.__id__ == prov_name), None) if provider is not None: data = provider.search(new_state.get("query", "")) - return Response( - json.dumps([asdict(x) for x in data]), mimetype="application/json" - ) + return make_response(jsonify([asdict(x) for x in data])) return "" @@ -130,7 +127,7 @@ def metadata_search(): locale = get_locale() if query: static_cover = url_for("static", filename="generic_cover.jpg") - # start = current_milli_time() + # ret = cl[0].search(query, static_cover, locale) with concurrent.futures.ThreadPoolExecutor(max_workers=5) as executor: meta = { executor.submit(c.search, query, static_cover, locale): c @@ -139,5 +136,4 @@ def metadata_search(): } for future in concurrent.futures.as_completed(meta): data.extend([asdict(x) for x in future.result() if x]) - # log.info({'Time elapsed {}'.format(current_milli_time()-start)}) - return Response(json.dumps(data), mimetype="application/json") + return make_response(jsonify(data)) diff --git a/cps/services/background_scheduler.py b/cps/services/background_scheduler.py index 72991559..55b12df3 100644 --- a/cps/services/background_scheduler.py +++ b/cps/services/background_scheduler.py @@ -42,6 +42,7 @@ class BackgroundScheduler: if cls._instance is None: cls._instance = super(BackgroundScheduler, cls).__new__(cls) cls.log = logger.create() + logger.logging.getLogger('tzlocal').setLevel(logger.logging.WARNING) cls.scheduler = BScheduler() cls.scheduler.start() diff --git a/cps/services/gmail.py b/cps/services/gmail.py index 5b0cdbe5..67b346a8 100644 --- a/cps/services/gmail.py +++ b/cps/services/gmail.py @@ -92,7 +92,7 @@ def send_messsage(token, msg): if creds and creds.expired and creds.refresh_token: creds.refresh(Request()) service = build('gmail', 'v1', credentials=creds) - message_as_bytes = msg.as_bytes() # the message should converted from string to bytes. + message_as_bytes = msg.as_bytes() # the message should be converted from string to bytes. message_as_base64 = base64.urlsafe_b64encode(message_as_bytes) # encode in base64 (printable letters coding) raw = message_as_base64.decode() # convert to something JSON serializable body = {'raw': raw} diff --git a/cps/services/goodreads_support.py b/cps/services/goodreads_support.py index b3b7bdef..c3bdbacb 100644 --- a/cps/services/goodreads_support.py +++ b/cps/services/goodreads_support.py @@ -117,7 +117,7 @@ def get_author_info(author_name): def get_other_books(author_info, library_books=None): - # Get all identifiers (ISBN, Goodreads, etc) and filter author's books by that list so we show fewer duplicates + # Get all identifiers (ISBN, Goodreads, etc.) and filter author's books by that list so we show fewer duplicates # Note: Not all images will be shown, even though they're available on Goodreads.com. # See https://www.goodreads.com/topic/show/18213769-goodreads-book-images diff --git a/cps/services/worker.py b/cps/services/worker.py index dce3da79..a804f633 100644 --- a/cps/services/worker.py +++ b/cps/services/worker.py @@ -199,7 +199,7 @@ class CalibreTask: self.run(*args) except Exception as ex: self._handleError(str(ex)) - log.error_or_exception(ex) + log.exception(ex) self.end_time = datetime.now() @@ -235,7 +235,7 @@ class CalibreTask: @property def dead(self): - """Determines whether or not this task can be garbage collected + """Determines whether this task can be garbage collected We have a separate dictating this because there may be certain tasks that want to override this """ diff --git a/cps/shelf.py b/cps/shelf.py index c8f43446..e4f93909 100644 --- a/cps/shelf.py +++ b/cps/shelf.py @@ -21,7 +21,7 @@ # along with this program. If not, see . import sys -from datetime import datetime +from datetime import datetime, timezone from flask import Blueprint, flash, redirect, request, url_for, abort from flask_babel import gettext as _ @@ -80,7 +80,7 @@ def add_to_shelf(shelf_id, book_id): return "%s is a invalid Book Id. Could not be added to Shelf" % book_id, 400 shelf.books.append(ub.BookShelf(shelf=shelf.id, book_id=book_id, order=maxOrder + 1)) - shelf.last_modified = datetime.utcnow() + shelf.last_modified = datetime.now(timezone.utc) try: ub.session.merge(shelf) ub.session.commit() @@ -139,7 +139,7 @@ def search_to_shelf(shelf_id): for book in books_for_shelf: maxOrder += 1 shelf.books.append(ub.BookShelf(shelf=shelf.id, book_id=book, order=maxOrder)) - shelf.last_modified = datetime.utcnow() + shelf.last_modified = datetime.now(timezone.utc) try: ub.session.merge(shelf) ub.session.commit() @@ -185,7 +185,7 @@ def remove_from_shelf(shelf_id, book_id): try: ub.session.delete(book_shelf) - shelf.last_modified = datetime.utcnow() + shelf.last_modified = datetime.now(timezone.utc) ub.session.commit() except (OperationalError, InvalidRequestError) as e: ub.session.rollback() @@ -250,7 +250,7 @@ def show_simpleshelf(shelf_id): return render_show_shelf(2, shelf_id, 1, None) -@shelf.route("/shelf/", defaults={"sort_param": "order", 'page': 1}) +@shelf.route("/shelf/", defaults={"sort_param": "stored", 'page': 1}) @shelf.route("/shelf//", defaults={'page': 1}) @shelf.route("/shelf///") @login_required_if_no_ano @@ -271,7 +271,7 @@ def order_shelf(shelf_id): for book in books_in_shelf: setattr(book, 'order', to_save[str(book.book_id)]) counter += 1 - # if order different from before -> shelf.last_modified = datetime.utcnow() + # if order different from before -> shelf.last_modified = datetime.now(timezone.utc) try: ub.session.commit() except (OperationalError, InvalidRequestError) as e: @@ -418,29 +418,37 @@ def change_shelf_order(shelf_id, order): def render_show_shelf(shelf_type, shelf_id, page_no, sort_param): shelf = ub.session.query(ub.Shelf).filter(ub.Shelf.id == shelf_id).first() - + status = current_user.get_view_property("shelf", 'man') # check user is allowed to access shelf if shelf and check_shelf_view_permissions(shelf): if shelf_type == 1: - # order = [ub.BookShelf.order.asc()] - if sort_param == 'pubnew': - change_shelf_order(shelf_id, [db.Books.pubdate.desc()]) - if sort_param == 'pubold': - change_shelf_order(shelf_id, [db.Books.pubdate]) - if sort_param == 'abc': - change_shelf_order(shelf_id, [db.Books.sort]) - if sort_param == 'zyx': - change_shelf_order(shelf_id, [db.Books.sort.desc()]) - if sort_param == 'new': - change_shelf_order(shelf_id, [db.Books.timestamp.desc()]) - if sort_param == 'old': - change_shelf_order(shelf_id, [db.Books.timestamp]) - if sort_param == 'authaz': - change_shelf_order(shelf_id, [db.Books.author_sort.asc(), db.Series.name, db.Books.series_index]) - if sort_param == 'authza': - change_shelf_order(shelf_id, [db.Books.author_sort.desc(), - db.Series.name.desc(), - db.Books.series_index.desc()]) + if status != 'on': + if sort_param == 'stored': + sort_param = current_user.get_view_property("shelf", 'stored') + else: + current_user.set_view_property("shelf", 'stored', sort_param) + if sort_param == 'pubnew': + change_shelf_order(shelf_id, [db.Books.pubdate.desc()]) + if sort_param == 'pubold': + change_shelf_order(shelf_id, [db.Books.pubdate]) + if sort_param == 'shelfnew': + change_shelf_order(shelf_id, [ub.BookShelf.date_added.desc()]) + if sort_param == 'shelfold': + change_shelf_order(shelf_id, [ub.BookShelf.date_added]) + if sort_param == 'abc': + change_shelf_order(shelf_id, [db.Books.sort]) + if sort_param == 'zyx': + change_shelf_order(shelf_id, [db.Books.sort.desc()]) + if sort_param == 'new': + change_shelf_order(shelf_id, [db.Books.timestamp.desc()]) + if sort_param == 'old': + change_shelf_order(shelf_id, [db.Books.timestamp]) + if sort_param == 'authaz': + change_shelf_order(shelf_id, [db.Books.author_sort.asc(), db.Series.name, db.Books.series_index]) + if sort_param == 'authza': + change_shelf_order(shelf_id, [db.Books.author_sort.desc(), + db.Series.name.desc(), + db.Books.series_index.desc()]) page = "shelf.html" pagesize = 0 else: @@ -453,7 +461,7 @@ def render_show_shelf(shelf_type, shelf_id, page_no, sort_param): [ub.BookShelf.order.asc()], True, config.config_read_column, ub.BookShelf, ub.BookShelf.book_id == db.Books.id) - # delete chelf entries where book is not existent anymore, can happen if book is deleted outside calibre-web + # delete shelf entries where book is not existent anymore, can happen if book is deleted outside calibre-web wrong_entries = calibre_db.session.query(ub.BookShelf) \ .join(db.Books, ub.BookShelf.book_id == db.Books.id, isouter=True) \ .filter(db.Books.id == None).all() @@ -472,7 +480,9 @@ def render_show_shelf(shelf_type, shelf_id, page_no, sort_param): pagination=pagination, title=_("Shelf: '%(name)s'", name=shelf.name), shelf=shelf, - page="shelf") + page="shelf", + status=status, + order=sort_param) else: flash(_("Error opening shelf. Shelf does not exist or is not accessible"), category="error") return redirect(url_for("web.index")) diff --git a/cps/static/css/basic.css b/cps/static/css/basic.css new file mode 100644 index 00000000..7fe9e5b8 --- /dev/null +++ b/cps/static/css/basic.css @@ -0,0 +1,108 @@ +body { + margin: 0; +} + +nav { + height: 75px; + padding: 5px 20px; + width: 100%; + display: table; + box-sizing: border-box; + border-bottom: 1px solid black; +} + +nav > * { + display: inline-block; + display: table-cell; + vertical-align: middle; + float: none; + text-align: center; + width: auto; +} + +nav > *:first-child { + text-align: left; + width: 1%; +} + +.theme{ + text-align: center; + margin: 10px; +} +nav > *:last-child { + text-align: right; + width: 1%; +} + +nav > a { + color: black; + margin: 0 20px; +} + +.search { + margin: auto auto; +} + +form > input { + width: 18ch; + padding-left: 4px; +} + +form > * { + height: 50px; + background-color: white; + border-radius: 0; + border: 1px solid #ccc; + padding: 0; + margin: 0; + display: inline-block; + vertical-align: top; +} + +form > span { + margin-left: -5px; +} + +button { + border: none; + padding: 0 10px; + margin: 0; + width: 160px; + height: 100%; + background-color: white; +} + +.body { + padding: 5px 20px; +} + +a { + color: black; +} + +img { + width: 150px; + height: 250px; + object-fit: cover; +} + +.listing { + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; + margin-right: 20px; +} + +.pagination { + padding: 10px 0; + height: 20px; + font-weight: 700; +} + +.pagination > div { + float: left; +} + +.pagination > div:last-child { + float: right; +} diff --git a/cps/static/css/caliBlur.css b/cps/static/css/caliBlur.css index 3647b871..c5e96bf4 100644 --- a/cps/static/css/caliBlur.css +++ b/cps/static/css/caliBlur.css @@ -3268,6 +3268,10 @@ div.btn-group[role=group][aria-label="Download, send to Kindle, reading"] > div. left: auto !important } +ul.dropdown-menu.offscreen { + margin: 2px -160px 0 +} + div.btn-group[role=group][aria-label="Download, send to Kindle, reading"] .dropdown-menu.offscreen { position: fixed; top: 120px; @@ -4333,6 +4337,7 @@ body.advanced_search > div.container-fluid > div.row-fluid > div.col-sm-10 > div .navbar-right > li > ul.dropdown-menu.offscreen { right: -10px + } .login .plexBack, body.login > div.container-fluid > div.row-fluid > div.col-sm-2, body.login > div.navbar.navbar-default.navbar-static-top > div > form { @@ -7148,12 +7153,11 @@ body.edituser.admin > div.container-fluid > div.row-fluid > div.col-sm-10 > div. } body.editbook > div.container-fluid > div.row-fluid > div.col-sm-10 > div.col-sm-3, body.upload > div.container-fluid > div.row-fluid > div.col-sm-10 > div.col-sm-3 { - max-width: 130px; - width: 130px; - height: 180px; - margin: 0; + position: relative; + max-width: unset; + width: 100%; + height: unset; padding: 15px; - position: absolute } body.editbook > div.container-fluid > div.row-fluid > div.col-sm-10 > form > div.col-sm-9, body.upload > div.container-fluid > div.row-fluid > div.col-sm-10 > form > div.col-sm-9 { @@ -7162,10 +7166,6 @@ body.edituser.admin > div.container-fluid > div.row-fluid > div.col-sm-10 > div. width: 100% } - body.editbook > div.container-fluid > div.row-fluid > div.col-sm-10 > form > div.col-sm-9 > .form-group:nth-child(1), body.editbook > div.container-fluid > div.row-fluid > div.col-sm-10 > form > div.col-sm-9 > .form-group:nth-child(2), body.upload > div.container-fluid > div.row-fluid > div.col-sm-10 > form > div.col-sm-9 > .form-group:nth-child(1), body.upload > div.container-fluid > div.row-fluid > div.col-sm-10 > form > div.col-sm-9 > .form-group:nth-child(2) { - padding-left: 120px - } - #deleteButton, body.editbook > div.container-fluid > div.row-fluid > div.col-sm-10 > div.col-sm-3 > div.text-center > #delete, body.upload > div.container-fluid > div.row-fluid > div.col-sm-10 > div.col-sm-3 > div.text-center > #delete { top: 48px; height: 42px @@ -7951,3 +7951,5 @@ div.comments[data-readmore] { transition: height 300ms; overflow: hidden } + +.dropdown-menu > .offscreen diff --git a/cps/static/css/epub_themes.css b/cps/static/css/epub_themes.css index 16c27805..b9cbcedb 100644 --- a/cps/static/css/epub_themes.css +++ b/cps/static/css/epub_themes.css @@ -15,5 +15,5 @@ .blackTheme { background: #000; - color: #fff + color: #fff; } \ No newline at end of file diff --git a/cps/static/css/libs/typeahead.css b/cps/static/css/libs/typeahead.css index fcc17a5b..b00303fe 100644 --- a/cps/static/css/libs/typeahead.css +++ b/cps/static/css/libs/typeahead.css @@ -157,7 +157,6 @@ fieldset[disabled] .twitter-typeahead .tt-input { list-style: none; font-size: 14px; background-color: #ffffff; - border: 1px solid #cccccc; border: 1px solid rgba(0, 0, 0, 0.15); border-radius: 4px; -webkit-box-shadow: 0 6px 12px rgba(0, 0, 0, 0.175); diff --git a/cps/static/css/main.css b/cps/static/css/main.css index 0dc0532c..571f546c 100644 --- a/cps/static/css/main.css +++ b/cps/static/css/main.css @@ -77,7 +77,6 @@ body { } #panels a { - visibility: hidden; width: 18px; height: 20px; overflow: hidden; @@ -512,12 +511,12 @@ input:-moz-placeholder { color: #454545; } position: fixed; top: 50%; left: 50%; - width: 630px; + transform: translate(-50%, -50%); + width: 100vw; height: auto; + max-width: 630px; z-index: 2000; visibility: hidden; - margin-left: -320px; - margin-top: -160px; } .overlay { diff --git a/cps/static/js/caliBlur.js b/cps/static/js/caliBlur.js old mode 100755 new mode 100644 diff --git a/cps/static/js/edit_books.js b/cps/static/js/edit_books.js index c1eb319d..54f615fb 100644 --- a/cps/static/js/edit_books.js +++ b/cps/static/js/edit_books.js @@ -3,9 +3,9 @@ */ /* global Bloodhound, language, Modernizr, tinymce, getPath */ -if ($("#description").length) { +if ($("#comments").length && typeof tinymce !== "undefined") { tinymce.init({ - selector: "#description", + selector: "#comments", plugins: 'code', branding: false, menubar: "edit view format", @@ -93,7 +93,7 @@ var authors = new Bloodhound({ }, }); -$(".form-group #bookAuthor").typeahead( +$(".form-group #authors").typeahead( { highlight: true, minLength: 1, @@ -243,13 +243,13 @@ $("#search").on("change input.typeahead:selected", function(event) { }); }); -$("#btn-upload-format").on("change", function () { +/*$("#btn-upload-format").on("change", function () { var filename = $(this).val(); if (filename.substring(3, 11) === "fakepath") { filename = filename.substring(12); } // Remove c:\fake at beginning from localhost chrome $("#upload-format").text(filename); -}); +});*/ $("#btn-upload-cover").on("change", function () { var filename = $(this).val(); @@ -261,8 +261,8 @@ $("#btn-upload-cover").on("change", function () { $("#xchange").click(function () { this.blur(); - var title = $("#book_title").val(); - $("#book_title").val($("#bookAuthor").val()); - $("#bookAuthor").val(title); + var title = $("#title").val(); + $("#title").val($("#authors").val()); + $("#authors").val(title); }); diff --git a/cps/static/js/get_meta.js b/cps/static/js/get_meta.js index 43a40fa6..df4119dc 100644 --- a/cps/static/js/get_meta.js +++ b/cps/static/js/get_meta.js @@ -38,12 +38,12 @@ $(function () { } function populateForm (book) { - tinymce.get("description").setContent(book.description); + tinymce.get("comments").setContent(book.description); var uniqueTags = getUniqueValues('tags', book) var uniqueLanguages = getUniqueValues('languages', book) var ampSeparatedAuthors = (book.authors || []).join(" & "); - $("#bookAuthor").val(ampSeparatedAuthors); - $("#book_title").val(book.title); + $("#authors").val(ampSeparatedAuthors); + $("#title").val(book.title); $("#tags").val(uniqueTags.join(", ")); $("#languages").val(uniqueLanguages.join(", ")); $("#rating").data("rating").setValue(Math.round(book.rating)); @@ -172,7 +172,7 @@ $(function () { $("#get_meta").click(function () { populate_provider(); - var bookTitle = $("#book_title").val(); + var bookTitle = $("#title").val(); $("#keyword").val(bookTitle); keyword = bookTitle; doSearch(bookTitle); diff --git a/cps/static/js/libs/bootstrap-datepicker/locales/bootstrap-datepicker.sl.min.js b/cps/static/js/libs/bootstrap-datepicker/locales/bootstrap-datepicker.sl.min.js new file mode 100644 index 00000000..831cf739 --- /dev/null +++ b/cps/static/js/libs/bootstrap-datepicker/locales/bootstrap-datepicker.sl.min.js @@ -0,0 +1 @@ +!function(a){a.fn.datepicker.dates.sl={days:["Nedelja","Ponedeljek","Torek","Sreda","Četrtek","Petek","Sobota"],daysShort:["Ned","Pon","Tor","Sre","Čet","Pet","Sob"],daysMin:["Ne","Po","To","Sr","Če","Pe","So"],months:["Januar","Februar","Marec","April","Maj","Junij","Julij","Avgust","September","Oktober","November","December"],monthsShort:["Jan","Feb","Mar","Apr","Maj","Jun","Jul","Avg","Sep","Okt","Nov","Dec"],today:"Danes",weekStart:1}}(jQuery); \ No newline at end of file diff --git a/cps/static/js/libs/pdf.mjs b/cps/static/js/libs/pdf.js similarity index 80% rename from cps/static/js/libs/pdf.mjs rename to cps/static/js/libs/pdf.js index 7906d1a9..13face72 100644 --- a/cps/static/js/libs/pdf.mjs +++ b/cps/static/js/libs/pdf.js @@ -2,7 +2,7 @@ * @licstart The following is the entire license notice for the * JavaScript code in this page * - * Copyright 2023 Mozilla Foundation + * Copyright 2024 Mozilla Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,8 +20,4712 @@ * JavaScript code in this page */ -/******/ // The require scope -/******/ var __webpack_require__ = {}; +/******/ var __webpack_modules__ = ({ + +/***/ 9306: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); +var tryToString = __webpack_require__(6823); + +var $TypeError = TypeError; + +// `Assert: IsCallable(argument) is true` +module.exports = function (argument) { + if (isCallable(argument)) return argument; + throw new $TypeError(tryToString(argument) + ' is not a function'); +}; + + +/***/ }), + +/***/ 3506: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isPossiblePrototype = __webpack_require__(3925); + +var $String = String; +var $TypeError = TypeError; + +module.exports = function (argument) { + if (isPossiblePrototype(argument)) return argument; + throw new $TypeError("Can't set " + $String(argument) + ' as a prototype'); +}; + + +/***/ }), + +/***/ 7080: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var has = (__webpack_require__(4402).has); + +// Perform ? RequireInternalSlot(M, [[SetData]]) +module.exports = function (it) { + has(it); + return it; +}; + + +/***/ }), + +/***/ 679: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isPrototypeOf = __webpack_require__(1625); + +var $TypeError = TypeError; + +module.exports = function (it, Prototype) { + if (isPrototypeOf(Prototype, it)) return it; + throw new $TypeError('Incorrect invocation'); +}; + + +/***/ }), + +/***/ 8551: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isObject = __webpack_require__(34); + +var $String = String; +var $TypeError = TypeError; + +// `Assert: Type(argument) is Object` +module.exports = function (argument) { + if (isObject(argument)) return argument; + throw new $TypeError($String(argument) + ' is not an object'); +}; + + +/***/ }), + +/***/ 7811: +/***/ ((module) => { + + +// eslint-disable-next-line es/no-typed-arrays -- safe +module.exports = typeof ArrayBuffer != 'undefined' && typeof DataView != 'undefined'; + + +/***/ }), + +/***/ 7394: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThisAccessor = __webpack_require__(6706); +var classof = __webpack_require__(4576); + +var $TypeError = TypeError; + +// Includes +// - Perform ? RequireInternalSlot(O, [[ArrayBufferData]]). +// - If IsSharedArrayBuffer(O) is true, throw a TypeError exception. +module.exports = uncurryThisAccessor(ArrayBuffer.prototype, 'byteLength', 'get') || function (O) { + if (classof(O) !== 'ArrayBuffer') throw new $TypeError('ArrayBuffer expected'); + return O.byteLength; +}; + + +/***/ }), + +/***/ 3238: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var arrayBufferByteLength = __webpack_require__(7394); + +var slice = uncurryThis(ArrayBuffer.prototype.slice); + +module.exports = function (O) { + if (arrayBufferByteLength(O) !== 0) return false; + try { + slice(O, 0, 0); + return false; + } catch (error) { + return true; + } +}; + + +/***/ }), + +/***/ 5636: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var uncurryThis = __webpack_require__(9504); +var uncurryThisAccessor = __webpack_require__(6706); +var toIndex = __webpack_require__(7696); +var isDetached = __webpack_require__(3238); +var arrayBufferByteLength = __webpack_require__(7394); +var detachTransferable = __webpack_require__(4483); +var PROPER_STRUCTURED_CLONE_TRANSFER = __webpack_require__(1548); + +var structuredClone = global.structuredClone; +var ArrayBuffer = global.ArrayBuffer; +var DataView = global.DataView; +var TypeError = global.TypeError; +var min = Math.min; +var ArrayBufferPrototype = ArrayBuffer.prototype; +var DataViewPrototype = DataView.prototype; +var slice = uncurryThis(ArrayBufferPrototype.slice); +var isResizable = uncurryThisAccessor(ArrayBufferPrototype, 'resizable', 'get'); +var maxByteLength = uncurryThisAccessor(ArrayBufferPrototype, 'maxByteLength', 'get'); +var getInt8 = uncurryThis(DataViewPrototype.getInt8); +var setInt8 = uncurryThis(DataViewPrototype.setInt8); + +module.exports = (PROPER_STRUCTURED_CLONE_TRANSFER || detachTransferable) && function (arrayBuffer, newLength, preserveResizability) { + var byteLength = arrayBufferByteLength(arrayBuffer); + var newByteLength = newLength === undefined ? byteLength : toIndex(newLength); + var fixedLength = !isResizable || !isResizable(arrayBuffer); + var newBuffer; + if (isDetached(arrayBuffer)) throw new TypeError('ArrayBuffer is detached'); + if (PROPER_STRUCTURED_CLONE_TRANSFER) { + arrayBuffer = structuredClone(arrayBuffer, { transfer: [arrayBuffer] }); + if (byteLength === newByteLength && (preserveResizability || fixedLength)) return arrayBuffer; + } + if (byteLength >= newByteLength && (!preserveResizability || fixedLength)) { + newBuffer = slice(arrayBuffer, 0, newByteLength); + } else { + var options = preserveResizability && !fixedLength && maxByteLength ? { maxByteLength: maxByteLength(arrayBuffer) } : undefined; + newBuffer = new ArrayBuffer(newByteLength, options); + var a = new DataView(arrayBuffer); + var b = new DataView(newBuffer); + var copyLength = min(newByteLength, byteLength); + for (var i = 0; i < copyLength; i++) setInt8(b, i, getInt8(a, i)); + } + if (!PROPER_STRUCTURED_CLONE_TRANSFER) detachTransferable(arrayBuffer); + return newBuffer; +}; + + +/***/ }), + +/***/ 4644: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_ARRAY_BUFFER = __webpack_require__(7811); +var DESCRIPTORS = __webpack_require__(3724); +var global = __webpack_require__(4475); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var hasOwn = __webpack_require__(9297); +var classof = __webpack_require__(6955); +var tryToString = __webpack_require__(6823); +var createNonEnumerableProperty = __webpack_require__(6699); +var defineBuiltIn = __webpack_require__(6840); +var defineBuiltInAccessor = __webpack_require__(2106); +var isPrototypeOf = __webpack_require__(1625); +var getPrototypeOf = __webpack_require__(2787); +var setPrototypeOf = __webpack_require__(2967); +var wellKnownSymbol = __webpack_require__(8227); +var uid = __webpack_require__(3392); +var InternalStateModule = __webpack_require__(1181); + +var enforceInternalState = InternalStateModule.enforce; +var getInternalState = InternalStateModule.get; +var Int8Array = global.Int8Array; +var Int8ArrayPrototype = Int8Array && Int8Array.prototype; +var Uint8ClampedArray = global.Uint8ClampedArray; +var Uint8ClampedArrayPrototype = Uint8ClampedArray && Uint8ClampedArray.prototype; +var TypedArray = Int8Array && getPrototypeOf(Int8Array); +var TypedArrayPrototype = Int8ArrayPrototype && getPrototypeOf(Int8ArrayPrototype); +var ObjectPrototype = Object.prototype; +var TypeError = global.TypeError; + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var TYPED_ARRAY_TAG = uid('TYPED_ARRAY_TAG'); +var TYPED_ARRAY_CONSTRUCTOR = 'TypedArrayConstructor'; +// Fixing native typed arrays in Opera Presto crashes the browser, see #595 +var NATIVE_ARRAY_BUFFER_VIEWS = NATIVE_ARRAY_BUFFER && !!setPrototypeOf && classof(global.opera) !== 'Opera'; +var TYPED_ARRAY_TAG_REQUIRED = false; +var NAME, Constructor, Prototype; + +var TypedArrayConstructorsList = { + Int8Array: 1, + Uint8Array: 1, + Uint8ClampedArray: 1, + Int16Array: 2, + Uint16Array: 2, + Int32Array: 4, + Uint32Array: 4, + Float32Array: 4, + Float64Array: 8 +}; + +var BigIntArrayConstructorsList = { + BigInt64Array: 8, + BigUint64Array: 8 +}; + +var isView = function isView(it) { + if (!isObject(it)) return false; + var klass = classof(it); + return klass === 'DataView' + || hasOwn(TypedArrayConstructorsList, klass) + || hasOwn(BigIntArrayConstructorsList, klass); +}; + +var getTypedArrayConstructor = function (it) { + var proto = getPrototypeOf(it); + if (!isObject(proto)) return; + var state = getInternalState(proto); + return (state && hasOwn(state, TYPED_ARRAY_CONSTRUCTOR)) ? state[TYPED_ARRAY_CONSTRUCTOR] : getTypedArrayConstructor(proto); +}; + +var isTypedArray = function (it) { + if (!isObject(it)) return false; + var klass = classof(it); + return hasOwn(TypedArrayConstructorsList, klass) + || hasOwn(BigIntArrayConstructorsList, klass); +}; + +var aTypedArray = function (it) { + if (isTypedArray(it)) return it; + throw new TypeError('Target is not a typed array'); +}; + +var aTypedArrayConstructor = function (C) { + if (isCallable(C) && (!setPrototypeOf || isPrototypeOf(TypedArray, C))) return C; + throw new TypeError(tryToString(C) + ' is not a typed array constructor'); +}; + +var exportTypedArrayMethod = function (KEY, property, forced, options) { + if (!DESCRIPTORS) return; + if (forced) for (var ARRAY in TypedArrayConstructorsList) { + var TypedArrayConstructor = global[ARRAY]; + if (TypedArrayConstructor && hasOwn(TypedArrayConstructor.prototype, KEY)) try { + delete TypedArrayConstructor.prototype[KEY]; + } catch (error) { + // old WebKit bug - some methods are non-configurable + try { + TypedArrayConstructor.prototype[KEY] = property; + } catch (error2) { /* empty */ } + } + } + if (!TypedArrayPrototype[KEY] || forced) { + defineBuiltIn(TypedArrayPrototype, KEY, forced ? property + : NATIVE_ARRAY_BUFFER_VIEWS && Int8ArrayPrototype[KEY] || property, options); + } +}; + +var exportTypedArrayStaticMethod = function (KEY, property, forced) { + var ARRAY, TypedArrayConstructor; + if (!DESCRIPTORS) return; + if (setPrototypeOf) { + if (forced) for (ARRAY in TypedArrayConstructorsList) { + TypedArrayConstructor = global[ARRAY]; + if (TypedArrayConstructor && hasOwn(TypedArrayConstructor, KEY)) try { + delete TypedArrayConstructor[KEY]; + } catch (error) { /* empty */ } + } + if (!TypedArray[KEY] || forced) { + // V8 ~ Chrome 49-50 `%TypedArray%` methods are non-writable non-configurable + try { + return defineBuiltIn(TypedArray, KEY, forced ? property : NATIVE_ARRAY_BUFFER_VIEWS && TypedArray[KEY] || property); + } catch (error) { /* empty */ } + } else return; + } + for (ARRAY in TypedArrayConstructorsList) { + TypedArrayConstructor = global[ARRAY]; + if (TypedArrayConstructor && (!TypedArrayConstructor[KEY] || forced)) { + defineBuiltIn(TypedArrayConstructor, KEY, property); + } + } +}; + +for (NAME in TypedArrayConstructorsList) { + Constructor = global[NAME]; + Prototype = Constructor && Constructor.prototype; + if (Prototype) enforceInternalState(Prototype)[TYPED_ARRAY_CONSTRUCTOR] = Constructor; + else NATIVE_ARRAY_BUFFER_VIEWS = false; +} + +for (NAME in BigIntArrayConstructorsList) { + Constructor = global[NAME]; + Prototype = Constructor && Constructor.prototype; + if (Prototype) enforceInternalState(Prototype)[TYPED_ARRAY_CONSTRUCTOR] = Constructor; +} + +// WebKit bug - typed arrays constructors prototype is Object.prototype +if (!NATIVE_ARRAY_BUFFER_VIEWS || !isCallable(TypedArray) || TypedArray === Function.prototype) { + // eslint-disable-next-line no-shadow -- safe + TypedArray = function TypedArray() { + throw new TypeError('Incorrect invocation'); + }; + if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) { + if (global[NAME]) setPrototypeOf(global[NAME], TypedArray); + } +} + +if (!NATIVE_ARRAY_BUFFER_VIEWS || !TypedArrayPrototype || TypedArrayPrototype === ObjectPrototype) { + TypedArrayPrototype = TypedArray.prototype; + if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) { + if (global[NAME]) setPrototypeOf(global[NAME].prototype, TypedArrayPrototype); + } +} + +// WebKit bug - one more object in Uint8ClampedArray prototype chain +if (NATIVE_ARRAY_BUFFER_VIEWS && getPrototypeOf(Uint8ClampedArrayPrototype) !== TypedArrayPrototype) { + setPrototypeOf(Uint8ClampedArrayPrototype, TypedArrayPrototype); +} + +if (DESCRIPTORS && !hasOwn(TypedArrayPrototype, TO_STRING_TAG)) { + TYPED_ARRAY_TAG_REQUIRED = true; + defineBuiltInAccessor(TypedArrayPrototype, TO_STRING_TAG, { + configurable: true, + get: function () { + return isObject(this) ? this[TYPED_ARRAY_TAG] : undefined; + } + }); + for (NAME in TypedArrayConstructorsList) if (global[NAME]) { + createNonEnumerableProperty(global[NAME], TYPED_ARRAY_TAG, NAME); + } +} + +module.exports = { + NATIVE_ARRAY_BUFFER_VIEWS: NATIVE_ARRAY_BUFFER_VIEWS, + TYPED_ARRAY_TAG: TYPED_ARRAY_TAG_REQUIRED && TYPED_ARRAY_TAG, + aTypedArray: aTypedArray, + aTypedArrayConstructor: aTypedArrayConstructor, + exportTypedArrayMethod: exportTypedArrayMethod, + exportTypedArrayStaticMethod: exportTypedArrayStaticMethod, + getTypedArrayConstructor: getTypedArrayConstructor, + isView: isView, + isTypedArray: isTypedArray, + TypedArray: TypedArray, + TypedArrayPrototype: TypedArrayPrototype +}; + + +/***/ }), + +/***/ 5370: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var lengthOfArrayLike = __webpack_require__(6198); + +module.exports = function (Constructor, list, $length) { + var index = 0; + var length = arguments.length > 2 ? $length : lengthOfArrayLike(list); + var result = new Constructor(length); + while (length > index) result[index] = list[index++]; + return result; +}; + + +/***/ }), + +/***/ 9617: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIndexedObject = __webpack_require__(5397); +var toAbsoluteIndex = __webpack_require__(5610); +var lengthOfArrayLike = __webpack_require__(6198); + +// `Array.prototype.{ indexOf, includes }` methods implementation +var createMethod = function (IS_INCLUDES) { + return function ($this, el, fromIndex) { + var O = toIndexedObject($this); + var length = lengthOfArrayLike(O); + if (length === 0) return !IS_INCLUDES && -1; + var index = toAbsoluteIndex(fromIndex, length); + var value; + // Array#includes uses SameValueZero equality algorithm + // eslint-disable-next-line no-self-compare -- NaN check + if (IS_INCLUDES && el !== el) while (length > index) { + value = O[index++]; + // eslint-disable-next-line no-self-compare -- NaN check + if (value !== value) return true; + // Array#indexOf ignores holes, Array#includes - not + } else for (;length > index; index++) { + if ((IS_INCLUDES || index in O) && O[index] === el) return IS_INCLUDES || index || 0; + } return !IS_INCLUDES && -1; + }; +}; + +module.exports = { + // `Array.prototype.includes` method + // https://tc39.es/ecma262/#sec-array.prototype.includes + includes: createMethod(true), + // `Array.prototype.indexOf` method + // https://tc39.es/ecma262/#sec-array.prototype.indexof + indexOf: createMethod(false) +}; + + +/***/ }), + +/***/ 4527: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var isArray = __webpack_require__(4376); + +var $TypeError = TypeError; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; + +// Safari < 13 does not throw an error in this case +var SILENT_ON_NON_WRITABLE_LENGTH_SET = DESCRIPTORS && !function () { + // makes no sense without proper strict mode support + if (this !== undefined) return true; + try { + // eslint-disable-next-line es/no-object-defineproperty -- safe + Object.defineProperty([], 'length', { writable: false }).length = 1; + } catch (error) { + return error instanceof TypeError; + } +}(); + +module.exports = SILENT_ON_NON_WRITABLE_LENGTH_SET ? function (O, length) { + if (isArray(O) && !getOwnPropertyDescriptor(O, 'length').writable) { + throw new $TypeError('Cannot set read only .length'); + } return O.length = length; +} : function (O, length) { + return O.length = length; +}; + + +/***/ }), + +/***/ 7628: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var lengthOfArrayLike = __webpack_require__(6198); + +// https://tc39.es/proposal-change-array-by-copy/#sec-array.prototype.toReversed +// https://tc39.es/proposal-change-array-by-copy/#sec-%typedarray%.prototype.toReversed +module.exports = function (O, C) { + var len = lengthOfArrayLike(O); + var A = new C(len); + var k = 0; + for (; k < len; k++) A[k] = O[len - k - 1]; + return A; +}; + + +/***/ }), + +/***/ 9928: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var lengthOfArrayLike = __webpack_require__(6198); +var toIntegerOrInfinity = __webpack_require__(1291); + +var $RangeError = RangeError; + +// https://tc39.es/proposal-change-array-by-copy/#sec-array.prototype.with +// https://tc39.es/proposal-change-array-by-copy/#sec-%typedarray%.prototype.with +module.exports = function (O, C, index, value) { + var len = lengthOfArrayLike(O); + var relativeIndex = toIntegerOrInfinity(index); + var actualIndex = relativeIndex < 0 ? len + relativeIndex : relativeIndex; + if (actualIndex >= len || actualIndex < 0) throw new $RangeError('Incorrect index'); + var A = new C(len); + var k = 0; + for (; k < len; k++) A[k] = k === actualIndex ? value : O[k]; + return A; +}; + + +/***/ }), + +/***/ 6319: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var anObject = __webpack_require__(8551); +var iteratorClose = __webpack_require__(9539); + +// call something on iterator step with safe closing on error +module.exports = function (iterator, fn, value, ENTRIES) { + try { + return ENTRIES ? fn(anObject(value)[0], value[1]) : fn(value); + } catch (error) { + iteratorClose(iterator, 'throw', error); + } +}; + + +/***/ }), + +/***/ 4576: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +var toString = uncurryThis({}.toString); +var stringSlice = uncurryThis(''.slice); + +module.exports = function (it) { + return stringSlice(toString(it), 8, -1); +}; + + +/***/ }), + +/***/ 6955: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var TO_STRING_TAG_SUPPORT = __webpack_require__(2140); +var isCallable = __webpack_require__(4901); +var classofRaw = __webpack_require__(4576); +var wellKnownSymbol = __webpack_require__(8227); + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var $Object = Object; + +// ES3 wrong here +var CORRECT_ARGUMENTS = classofRaw(function () { return arguments; }()) === 'Arguments'; + +// fallback for IE11 Script Access Denied error +var tryGet = function (it, key) { + try { + return it[key]; + } catch (error) { /* empty */ } +}; + +// getting tag from ES6+ `Object.prototype.toString` +module.exports = TO_STRING_TAG_SUPPORT ? classofRaw : function (it) { + var O, tag, result; + return it === undefined ? 'Undefined' : it === null ? 'Null' + // @@toStringTag case + : typeof (tag = tryGet(O = $Object(it), TO_STRING_TAG)) == 'string' ? tag + // builtinTag case + : CORRECT_ARGUMENTS ? classofRaw(O) + // ES3 arguments fallback + : (result = classofRaw(O)) === 'Object' && isCallable(O.callee) ? 'Arguments' : result; +}; + + +/***/ }), + +/***/ 7740: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var hasOwn = __webpack_require__(9297); +var ownKeys = __webpack_require__(5031); +var getOwnPropertyDescriptorModule = __webpack_require__(7347); +var definePropertyModule = __webpack_require__(4913); + +module.exports = function (target, source, exceptions) { + var keys = ownKeys(source); + var defineProperty = definePropertyModule.f; + var getOwnPropertyDescriptor = getOwnPropertyDescriptorModule.f; + for (var i = 0; i < keys.length; i++) { + var key = keys[i]; + if (!hasOwn(target, key) && !(exceptions && hasOwn(exceptions, key))) { + defineProperty(target, key, getOwnPropertyDescriptor(source, key)); + } + } +}; + + +/***/ }), + +/***/ 2211: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); + +module.exports = !fails(function () { + function F() { /* empty */ } + F.prototype.constructor = null; + // eslint-disable-next-line es/no-object-getprototypeof -- required for testing + return Object.getPrototypeOf(new F()) !== F.prototype; +}); + + +/***/ }), + +/***/ 2529: +/***/ ((module) => { + + +// `CreateIterResultObject` abstract operation +// https://tc39.es/ecma262/#sec-createiterresultobject +module.exports = function (value, done) { + return { value: value, done: done }; +}; + + +/***/ }), + +/***/ 6699: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var definePropertyModule = __webpack_require__(4913); +var createPropertyDescriptor = __webpack_require__(6980); + +module.exports = DESCRIPTORS ? function (object, key, value) { + return definePropertyModule.f(object, key, createPropertyDescriptor(1, value)); +} : function (object, key, value) { + object[key] = value; + return object; +}; + + +/***/ }), + +/***/ 6980: +/***/ ((module) => { + + +module.exports = function (bitmap, value) { + return { + enumerable: !(bitmap & 1), + configurable: !(bitmap & 2), + writable: !(bitmap & 4), + value: value + }; +}; + + +/***/ }), + +/***/ 4659: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var definePropertyModule = __webpack_require__(4913); +var createPropertyDescriptor = __webpack_require__(6980); + +module.exports = function (object, key, value) { + if (DESCRIPTORS) definePropertyModule.f(object, key, createPropertyDescriptor(0, value)); + else object[key] = value; +}; + + +/***/ }), + +/***/ 2106: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var makeBuiltIn = __webpack_require__(283); +var defineProperty = __webpack_require__(4913); + +module.exports = function (target, name, descriptor) { + if (descriptor.get) makeBuiltIn(descriptor.get, name, { getter: true }); + if (descriptor.set) makeBuiltIn(descriptor.set, name, { setter: true }); + return defineProperty.f(target, name, descriptor); +}; + + +/***/ }), + +/***/ 6840: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); +var definePropertyModule = __webpack_require__(4913); +var makeBuiltIn = __webpack_require__(283); +var defineGlobalProperty = __webpack_require__(9433); + +module.exports = function (O, key, value, options) { + if (!options) options = {}; + var simple = options.enumerable; + var name = options.name !== undefined ? options.name : key; + if (isCallable(value)) makeBuiltIn(value, name, options); + if (options.global) { + if (simple) O[key] = value; + else defineGlobalProperty(key, value); + } else { + try { + if (!options.unsafe) delete O[key]; + else if (O[key]) simple = true; + } catch (error) { /* empty */ } + if (simple) O[key] = value; + else definePropertyModule.f(O, key, { + value: value, + enumerable: false, + configurable: !options.nonConfigurable, + writable: !options.nonWritable + }); + } return O; +}; + + +/***/ }), + +/***/ 6279: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var defineBuiltIn = __webpack_require__(6840); + +module.exports = function (target, src, options) { + for (var key in src) defineBuiltIn(target, key, src[key], options); + return target; +}; + + +/***/ }), + +/***/ 9433: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); + +// eslint-disable-next-line es/no-object-defineproperty -- safe +var defineProperty = Object.defineProperty; + +module.exports = function (key, value) { + try { + defineProperty(global, key, { value: value, configurable: true, writable: true }); + } catch (error) { + global[key] = value; + } return value; +}; + + +/***/ }), + +/***/ 3724: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); + +// Detect IE8's incomplete defineProperty implementation +module.exports = !fails(function () { + // eslint-disable-next-line es/no-object-defineproperty -- required for testing + return Object.defineProperty({}, 1, { get: function () { return 7; } })[1] !== 7; +}); + + +/***/ }), + +/***/ 4483: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var tryNodeRequire = __webpack_require__(9714); +var PROPER_STRUCTURED_CLONE_TRANSFER = __webpack_require__(1548); + +var structuredClone = global.structuredClone; +var $ArrayBuffer = global.ArrayBuffer; +var $MessageChannel = global.MessageChannel; +var detach = false; +var WorkerThreads, channel, buffer, $detach; + +if (PROPER_STRUCTURED_CLONE_TRANSFER) { + detach = function (transferable) { + structuredClone(transferable, { transfer: [transferable] }); + }; +} else if ($ArrayBuffer) try { + if (!$MessageChannel) { + WorkerThreads = tryNodeRequire('worker_threads'); + if (WorkerThreads) $MessageChannel = WorkerThreads.MessageChannel; + } + + if ($MessageChannel) { + channel = new $MessageChannel(); + buffer = new $ArrayBuffer(2); + + $detach = function (transferable) { + channel.port1.postMessage(null, [transferable]); + }; + + if (buffer.byteLength === 2) { + $detach(buffer); + if (buffer.byteLength === 0) detach = $detach; + } + } +} catch (error) { /* empty */ } + +module.exports = detach; + + +/***/ }), + +/***/ 4055: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var isObject = __webpack_require__(34); + +var document = global.document; +// typeof document.createElement is 'object' in old IE +var EXISTS = isObject(document) && isObject(document.createElement); + +module.exports = function (it) { + return EXISTS ? document.createElement(it) : {}; +}; + + +/***/ }), + +/***/ 6837: +/***/ ((module) => { + + +var $TypeError = TypeError; +var MAX_SAFE_INTEGER = 0x1FFFFFFFFFFFFF; // 2 ** 53 - 1 == 9007199254740991 + +module.exports = function (it) { + if (it > MAX_SAFE_INTEGER) throw $TypeError('Maximum allowed index exceeded'); + return it; +}; + + +/***/ }), + +/***/ 5002: +/***/ ((module) => { + + +module.exports = { + IndexSizeError: { s: 'INDEX_SIZE_ERR', c: 1, m: 1 }, + DOMStringSizeError: { s: 'DOMSTRING_SIZE_ERR', c: 2, m: 0 }, + HierarchyRequestError: { s: 'HIERARCHY_REQUEST_ERR', c: 3, m: 1 }, + WrongDocumentError: { s: 'WRONG_DOCUMENT_ERR', c: 4, m: 1 }, + InvalidCharacterError: { s: 'INVALID_CHARACTER_ERR', c: 5, m: 1 }, + NoDataAllowedError: { s: 'NO_DATA_ALLOWED_ERR', c: 6, m: 0 }, + NoModificationAllowedError: { s: 'NO_MODIFICATION_ALLOWED_ERR', c: 7, m: 1 }, + NotFoundError: { s: 'NOT_FOUND_ERR', c: 8, m: 1 }, + NotSupportedError: { s: 'NOT_SUPPORTED_ERR', c: 9, m: 1 }, + InUseAttributeError: { s: 'INUSE_ATTRIBUTE_ERR', c: 10, m: 1 }, + InvalidStateError: { s: 'INVALID_STATE_ERR', c: 11, m: 1 }, + SyntaxError: { s: 'SYNTAX_ERR', c: 12, m: 1 }, + InvalidModificationError: { s: 'INVALID_MODIFICATION_ERR', c: 13, m: 1 }, + NamespaceError: { s: 'NAMESPACE_ERR', c: 14, m: 1 }, + InvalidAccessError: { s: 'INVALID_ACCESS_ERR', c: 15, m: 1 }, + ValidationError: { s: 'VALIDATION_ERR', c: 16, m: 0 }, + TypeMismatchError: { s: 'TYPE_MISMATCH_ERR', c: 17, m: 1 }, + SecurityError: { s: 'SECURITY_ERR', c: 18, m: 1 }, + NetworkError: { s: 'NETWORK_ERR', c: 19, m: 1 }, + AbortError: { s: 'ABORT_ERR', c: 20, m: 1 }, + URLMismatchError: { s: 'URL_MISMATCH_ERR', c: 21, m: 1 }, + QuotaExceededError: { s: 'QUOTA_EXCEEDED_ERR', c: 22, m: 1 }, + TimeoutError: { s: 'TIMEOUT_ERR', c: 23, m: 1 }, + InvalidNodeTypeError: { s: 'INVALID_NODE_TYPE_ERR', c: 24, m: 1 }, + DataCloneError: { s: 'DATA_CLONE_ERR', c: 25, m: 1 } +}; + + +/***/ }), + +/***/ 7290: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var IS_DENO = __webpack_require__(516); +var IS_NODE = __webpack_require__(9088); + +module.exports = !IS_DENO && !IS_NODE + && typeof window == 'object' + && typeof document == 'object'; + + +/***/ }), + +/***/ 516: +/***/ ((module) => { + + +/* global Deno -- Deno case */ +module.exports = typeof Deno == 'object' && Deno && typeof Deno.version == 'object'; + + +/***/ }), + +/***/ 9088: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var classof = __webpack_require__(4576); + +module.exports = classof(global.process) === 'process'; + + +/***/ }), + +/***/ 9392: +/***/ ((module) => { + + +module.exports = typeof navigator != 'undefined' && String(navigator.userAgent) || ''; + + +/***/ }), + +/***/ 7388: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var userAgent = __webpack_require__(9392); + +var process = global.process; +var Deno = global.Deno; +var versions = process && process.versions || Deno && Deno.version; +var v8 = versions && versions.v8; +var match, version; + +if (v8) { + match = v8.split('.'); + // in old Chrome, versions of V8 isn't V8 = Chrome / 10 + // but their correct versions are not interesting for us + version = match[0] > 0 && match[0] < 4 ? 1 : +(match[0] + match[1]); +} + +// BrowserFS NodeJS `process` polyfill incorrectly set `.v8` to `0.0` +// so check `userAgent` even if `.v8` exists, but 0 +if (!version && userAgent) { + match = userAgent.match(/Edge\/(\d+)/); + if (!match || match[1] >= 74) { + match = userAgent.match(/Chrome\/(\d+)/); + if (match) version = +match[1]; + } +} + +module.exports = version; + + +/***/ }), + +/***/ 8727: +/***/ ((module) => { + + +// IE8- don't enum bug keys +module.exports = [ + 'constructor', + 'hasOwnProperty', + 'isPrototypeOf', + 'propertyIsEnumerable', + 'toLocaleString', + 'toString', + 'valueOf' +]; + + +/***/ }), + +/***/ 6193: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +var $Error = Error; +var replace = uncurryThis(''.replace); + +var TEST = (function (arg) { return String(new $Error(arg).stack); })('zxcasd'); +// eslint-disable-next-line redos/no-vulnerable -- safe +var V8_OR_CHAKRA_STACK_ENTRY = /\n\s*at [^:]*:[^\n]*/; +var IS_V8_OR_CHAKRA_STACK = V8_OR_CHAKRA_STACK_ENTRY.test(TEST); + +module.exports = function (stack, dropEntries) { + if (IS_V8_OR_CHAKRA_STACK && typeof stack == 'string' && !$Error.prepareStackTrace) { + while (dropEntries--) stack = replace(stack, V8_OR_CHAKRA_STACK_ENTRY, ''); + } return stack; +}; + + +/***/ }), + +/***/ 6518: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var getOwnPropertyDescriptor = (__webpack_require__(7347).f); +var createNonEnumerableProperty = __webpack_require__(6699); +var defineBuiltIn = __webpack_require__(6840); +var defineGlobalProperty = __webpack_require__(9433); +var copyConstructorProperties = __webpack_require__(7740); +var isForced = __webpack_require__(2796); + +/* + options.target - name of the target object + options.global - target is the global object + options.stat - export as static methods of target + options.proto - export as prototype methods of target + options.real - real prototype method for the `pure` version + options.forced - export even if the native feature is available + options.bind - bind methods to the target, required for the `pure` version + options.wrap - wrap constructors to preventing global pollution, required for the `pure` version + options.unsafe - use the simple assignment of property instead of delete + defineProperty + options.sham - add a flag to not completely full polyfills + options.enumerable - export as enumerable property + options.dontCallGetSet - prevent calling a getter on target + options.name - the .name of the function if it does not match the key +*/ +module.exports = function (options, source) { + var TARGET = options.target; + var GLOBAL = options.global; + var STATIC = options.stat; + var FORCED, target, key, targetProperty, sourceProperty, descriptor; + if (GLOBAL) { + target = global; + } else if (STATIC) { + target = global[TARGET] || defineGlobalProperty(TARGET, {}); + } else { + target = global[TARGET] && global[TARGET].prototype; + } + if (target) for (key in source) { + sourceProperty = source[key]; + if (options.dontCallGetSet) { + descriptor = getOwnPropertyDescriptor(target, key); + targetProperty = descriptor && descriptor.value; + } else targetProperty = target[key]; + FORCED = isForced(GLOBAL ? key : TARGET + (STATIC ? '.' : '#') + key, options.forced); + // contained in target + if (!FORCED && targetProperty !== undefined) { + if (typeof sourceProperty == typeof targetProperty) continue; + copyConstructorProperties(sourceProperty, targetProperty); + } + // add a flag to not completely full polyfills + if (options.sham || (targetProperty && targetProperty.sham)) { + createNonEnumerableProperty(sourceProperty, 'sham', true); + } + defineBuiltIn(target, key, sourceProperty, options); + } +}; + + +/***/ }), + +/***/ 9039: +/***/ ((module) => { + + +module.exports = function (exec) { + try { + return !!exec(); + } catch (error) { + return true; + } +}; + + +/***/ }), + +/***/ 6080: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(7476); +var aCallable = __webpack_require__(9306); +var NATIVE_BIND = __webpack_require__(616); + +var bind = uncurryThis(uncurryThis.bind); + +// optional / simple context binding +module.exports = function (fn, that) { + aCallable(fn); + return that === undefined ? fn : NATIVE_BIND ? bind(fn, that) : function (/* ...args */) { + return fn.apply(that, arguments); + }; +}; + + +/***/ }), + +/***/ 616: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); + +module.exports = !fails(function () { + // eslint-disable-next-line es/no-function-prototype-bind -- safe + var test = (function () { /* empty */ }).bind(); + // eslint-disable-next-line no-prototype-builtins -- safe + return typeof test != 'function' || test.hasOwnProperty('prototype'); +}); + + +/***/ }), + +/***/ 9565: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_BIND = __webpack_require__(616); + +var call = Function.prototype.call; + +module.exports = NATIVE_BIND ? call.bind(call) : function () { + return call.apply(call, arguments); +}; + + +/***/ }), + +/***/ 350: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var hasOwn = __webpack_require__(9297); + +var FunctionPrototype = Function.prototype; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getDescriptor = DESCRIPTORS && Object.getOwnPropertyDescriptor; + +var EXISTS = hasOwn(FunctionPrototype, 'name'); +// additional protection from minified / mangled / dropped function names +var PROPER = EXISTS && (function something() { /* empty */ }).name === 'something'; +var CONFIGURABLE = EXISTS && (!DESCRIPTORS || (DESCRIPTORS && getDescriptor(FunctionPrototype, 'name').configurable)); + +module.exports = { + EXISTS: EXISTS, + PROPER: PROPER, + CONFIGURABLE: CONFIGURABLE +}; + + +/***/ }), + +/***/ 6706: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var aCallable = __webpack_require__(9306); + +module.exports = function (object, key, method) { + try { + // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe + return uncurryThis(aCallable(Object.getOwnPropertyDescriptor(object, key)[method])); + } catch (error) { /* empty */ } +}; + + +/***/ }), + +/***/ 7476: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classofRaw = __webpack_require__(4576); +var uncurryThis = __webpack_require__(9504); + +module.exports = function (fn) { + // Nashorn bug: + // https://github.com/zloirock/core-js/issues/1128 + // https://github.com/zloirock/core-js/issues/1130 + if (classofRaw(fn) === 'Function') return uncurryThis(fn); +}; + + +/***/ }), + +/***/ 9504: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_BIND = __webpack_require__(616); + +var FunctionPrototype = Function.prototype; +var call = FunctionPrototype.call; +var uncurryThisWithBind = NATIVE_BIND && FunctionPrototype.bind.bind(call, call); + +module.exports = NATIVE_BIND ? uncurryThisWithBind : function (fn) { + return function () { + return call.apply(fn, arguments); + }; +}; + + +/***/ }), + +/***/ 7751: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var isCallable = __webpack_require__(4901); + +var aFunction = function (argument) { + return isCallable(argument) ? argument : undefined; +}; + +module.exports = function (namespace, method) { + return arguments.length < 2 ? aFunction(global[namespace]) : global[namespace] && global[namespace][method]; +}; + + +/***/ }), + +/***/ 1767: +/***/ ((module) => { + + +// `GetIteratorDirect(obj)` abstract operation +// https://tc39.es/proposal-iterator-helpers/#sec-getiteratordirect +module.exports = function (obj) { + return { + iterator: obj, + next: obj.next, + done: false + }; +}; + + +/***/ }), + +/***/ 8646: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var getIteratorMethod = __webpack_require__(851); + +module.exports = function (obj, stringHandling) { + if (!stringHandling || typeof obj !== 'string') anObject(obj); + var method = getIteratorMethod(obj); + return getIteratorDirect(anObject(method !== undefined ? call(method, obj) : obj)); +}; + + +/***/ }), + +/***/ 851: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(6955); +var getMethod = __webpack_require__(5966); +var isNullOrUndefined = __webpack_require__(4117); +var Iterators = __webpack_require__(6269); +var wellKnownSymbol = __webpack_require__(8227); + +var ITERATOR = wellKnownSymbol('iterator'); + +module.exports = function (it) { + if (!isNullOrUndefined(it)) return getMethod(it, ITERATOR) + || getMethod(it, '@@iterator') + || Iterators[classof(it)]; +}; + + +/***/ }), + +/***/ 81: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var tryToString = __webpack_require__(6823); +var getIteratorMethod = __webpack_require__(851); + +var $TypeError = TypeError; + +module.exports = function (argument, usingIterator) { + var iteratorMethod = arguments.length < 2 ? getIteratorMethod(argument) : usingIterator; + if (aCallable(iteratorMethod)) return anObject(call(iteratorMethod, argument)); + throw new $TypeError(tryToString(argument) + ' is not iterable'); +}; + + +/***/ }), + +/***/ 5966: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aCallable = __webpack_require__(9306); +var isNullOrUndefined = __webpack_require__(4117); + +// `GetMethod` abstract operation +// https://tc39.es/ecma262/#sec-getmethod +module.exports = function (V, P) { + var func = V[P]; + return isNullOrUndefined(func) ? undefined : aCallable(func); +}; + + +/***/ }), + +/***/ 3789: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var call = __webpack_require__(9565); +var toIntegerOrInfinity = __webpack_require__(1291); +var getIteratorDirect = __webpack_require__(1767); + +var INVALID_SIZE = 'Invalid size'; +var $RangeError = RangeError; +var $TypeError = TypeError; +var max = Math.max; + +var SetRecord = function (set, intSize) { + this.set = set; + this.size = max(intSize, 0); + this.has = aCallable(set.has); + this.keys = aCallable(set.keys); +}; + +SetRecord.prototype = { + getIterator: function () { + return getIteratorDirect(anObject(call(this.keys, this.set))); + }, + includes: function (it) { + return call(this.has, this.set, it); + } +}; + +// `GetSetRecord` abstract operation +// https://tc39.es/proposal-set-methods/#sec-getsetrecord +module.exports = function (obj) { + anObject(obj); + var numSize = +obj.size; + // NOTE: If size is undefined, then numSize will be NaN + // eslint-disable-next-line no-self-compare -- NaN check + if (numSize !== numSize) throw new $TypeError(INVALID_SIZE); + var intSize = toIntegerOrInfinity(numSize); + if (intSize < 0) throw new $RangeError(INVALID_SIZE); + return new SetRecord(obj, intSize); +}; + + +/***/ }), + +/***/ 4475: +/***/ (function(module) { + + +var check = function (it) { + return it && it.Math === Math && it; +}; + +// https://github.com/zloirock/core-js/issues/86#issuecomment-115759028 +module.exports = + // eslint-disable-next-line es/no-global-this -- safe + check(typeof globalThis == 'object' && globalThis) || + check(typeof window == 'object' && window) || + // eslint-disable-next-line no-restricted-globals -- safe + check(typeof self == 'object' && self) || + check(typeof global == 'object' && global) || + check(typeof this == 'object' && this) || + // eslint-disable-next-line no-new-func -- fallback + (function () { return this; })() || Function('return this')(); + + +/***/ }), + +/***/ 9297: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var toObject = __webpack_require__(8981); + +var hasOwnProperty = uncurryThis({}.hasOwnProperty); + +// `HasOwnProperty` abstract operation +// https://tc39.es/ecma262/#sec-hasownproperty +// eslint-disable-next-line es/no-object-hasown -- safe +module.exports = Object.hasOwn || function hasOwn(it, key) { + return hasOwnProperty(toObject(it), key); +}; + + +/***/ }), + +/***/ 421: +/***/ ((module) => { + + +module.exports = {}; + + +/***/ }), + +/***/ 397: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); + +module.exports = getBuiltIn('document', 'documentElement'); + + +/***/ }), + +/***/ 5917: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var fails = __webpack_require__(9039); +var createElement = __webpack_require__(4055); + +// Thanks to IE8 for its funny defineProperty +module.exports = !DESCRIPTORS && !fails(function () { + // eslint-disable-next-line es/no-object-defineproperty -- required for testing + return Object.defineProperty(createElement('div'), 'a', { + get: function () { return 7; } + }).a !== 7; +}); + + +/***/ }), + +/***/ 7055: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var fails = __webpack_require__(9039); +var classof = __webpack_require__(4576); + +var $Object = Object; +var split = uncurryThis(''.split); + +// fallback for non-array-like ES3 and non-enumerable old V8 strings +module.exports = fails(function () { + // throws an error in rhino, see https://github.com/mozilla/rhino/issues/346 + // eslint-disable-next-line no-prototype-builtins -- safe + return !$Object('z').propertyIsEnumerable(0); +}) ? function (it) { + return classof(it) === 'String' ? split(it, '') : $Object(it); +} : $Object; + + +/***/ }), + +/***/ 3167: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var setPrototypeOf = __webpack_require__(2967); + +// makes subclassing work correct for wrapped built-ins +module.exports = function ($this, dummy, Wrapper) { + var NewTarget, NewTargetPrototype; + if ( + // it can work only with native `setPrototypeOf` + setPrototypeOf && + // we haven't completely correct pre-ES6 way for getting `new.target`, so use this + isCallable(NewTarget = dummy.constructor) && + NewTarget !== Wrapper && + isObject(NewTargetPrototype = NewTarget.prototype) && + NewTargetPrototype !== Wrapper.prototype + ) setPrototypeOf($this, NewTargetPrototype); + return $this; +}; + + +/***/ }), + +/***/ 3706: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var isCallable = __webpack_require__(4901); +var store = __webpack_require__(7629); + +var functionToString = uncurryThis(Function.toString); + +// this helper broken in `core-js@3.4.1-3.4.4`, so we can't use `shared` helper +if (!isCallable(store.inspectSource)) { + store.inspectSource = function (it) { + return functionToString(it); + }; +} + +module.exports = store.inspectSource; + + +/***/ }), + +/***/ 1181: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_WEAK_MAP = __webpack_require__(8622); +var global = __webpack_require__(4475); +var isObject = __webpack_require__(34); +var createNonEnumerableProperty = __webpack_require__(6699); +var hasOwn = __webpack_require__(9297); +var shared = __webpack_require__(7629); +var sharedKey = __webpack_require__(6119); +var hiddenKeys = __webpack_require__(421); + +var OBJECT_ALREADY_INITIALIZED = 'Object already initialized'; +var TypeError = global.TypeError; +var WeakMap = global.WeakMap; +var set, get, has; + +var enforce = function (it) { + return has(it) ? get(it) : set(it, {}); +}; + +var getterFor = function (TYPE) { + return function (it) { + var state; + if (!isObject(it) || (state = get(it)).type !== TYPE) { + throw new TypeError('Incompatible receiver, ' + TYPE + ' required'); + } return state; + }; +}; + +if (NATIVE_WEAK_MAP || shared.state) { + var store = shared.state || (shared.state = new WeakMap()); + /* eslint-disable no-self-assign -- prototype methods protection */ + store.get = store.get; + store.has = store.has; + store.set = store.set; + /* eslint-enable no-self-assign -- prototype methods protection */ + set = function (it, metadata) { + if (store.has(it)) throw new TypeError(OBJECT_ALREADY_INITIALIZED); + metadata.facade = it; + store.set(it, metadata); + return metadata; + }; + get = function (it) { + return store.get(it) || {}; + }; + has = function (it) { + return store.has(it); + }; +} else { + var STATE = sharedKey('state'); + hiddenKeys[STATE] = true; + set = function (it, metadata) { + if (hasOwn(it, STATE)) throw new TypeError(OBJECT_ALREADY_INITIALIZED); + metadata.facade = it; + createNonEnumerableProperty(it, STATE, metadata); + return metadata; + }; + get = function (it) { + return hasOwn(it, STATE) ? it[STATE] : {}; + }; + has = function (it) { + return hasOwn(it, STATE); + }; +} + +module.exports = { + set: set, + get: get, + has: has, + enforce: enforce, + getterFor: getterFor +}; + + +/***/ }), + +/***/ 4209: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var wellKnownSymbol = __webpack_require__(8227); +var Iterators = __webpack_require__(6269); + +var ITERATOR = wellKnownSymbol('iterator'); +var ArrayPrototype = Array.prototype; + +// check on default Array iterator +module.exports = function (it) { + return it !== undefined && (Iterators.Array === it || ArrayPrototype[ITERATOR] === it); +}; + + +/***/ }), + +/***/ 4376: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(4576); + +// `IsArray` abstract operation +// https://tc39.es/ecma262/#sec-isarray +// eslint-disable-next-line es/no-array-isarray -- safe +module.exports = Array.isArray || function isArray(argument) { + return classof(argument) === 'Array'; +}; + + +/***/ }), + +/***/ 1108: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(6955); + +module.exports = function (it) { + var klass = classof(it); + return klass === 'BigInt64Array' || klass === 'BigUint64Array'; +}; + + +/***/ }), + +/***/ 4901: +/***/ ((module) => { + + +// https://tc39.es/ecma262/#sec-IsHTMLDDA-internal-slot +var documentAll = typeof document == 'object' && document.all; + +// `IsCallable` abstract operation +// https://tc39.es/ecma262/#sec-iscallable +// eslint-disable-next-line unicorn/no-typeof-undefined -- required for testing +module.exports = typeof documentAll == 'undefined' && documentAll !== undefined ? function (argument) { + return typeof argument == 'function' || argument === documentAll; +} : function (argument) { + return typeof argument == 'function'; +}; + + +/***/ }), + +/***/ 2796: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); +var isCallable = __webpack_require__(4901); + +var replacement = /#|\.prototype\./; + +var isForced = function (feature, detection) { + var value = data[normalize(feature)]; + return value === POLYFILL ? true + : value === NATIVE ? false + : isCallable(detection) ? fails(detection) + : !!detection; +}; + +var normalize = isForced.normalize = function (string) { + return String(string).replace(replacement, '.').toLowerCase(); +}; + +var data = isForced.data = {}; +var NATIVE = isForced.NATIVE = 'N'; +var POLYFILL = isForced.POLYFILL = 'P'; + +module.exports = isForced; + + +/***/ }), + +/***/ 4117: +/***/ ((module) => { + + +// we can't use just `it == null` since of `document.all` special case +// https://tc39.es/ecma262/#sec-IsHTMLDDA-internal-slot-aec +module.exports = function (it) { + return it === null || it === undefined; +}; + + +/***/ }), + +/***/ 34: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); + +module.exports = function (it) { + return typeof it == 'object' ? it !== null : isCallable(it); +}; + + +/***/ }), + +/***/ 3925: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isObject = __webpack_require__(34); + +module.exports = function (argument) { + return isObject(argument) || argument === null; +}; + + +/***/ }), + +/***/ 6395: +/***/ ((module) => { + + +module.exports = false; + + +/***/ }), + +/***/ 757: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); +var isCallable = __webpack_require__(4901); +var isPrototypeOf = __webpack_require__(1625); +var USE_SYMBOL_AS_UID = __webpack_require__(7040); + +var $Object = Object; + +module.exports = USE_SYMBOL_AS_UID ? function (it) { + return typeof it == 'symbol'; +} : function (it) { + var $Symbol = getBuiltIn('Symbol'); + return isCallable($Symbol) && isPrototypeOf($Symbol.prototype, $Object(it)); +}; + + +/***/ }), + +/***/ 507: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); + +module.exports = function (record, fn, ITERATOR_INSTEAD_OF_RECORD) { + var iterator = ITERATOR_INSTEAD_OF_RECORD ? record : record.iterator; + var next = record.next; + var step, result; + while (!(step = call(next, iterator)).done) { + result = fn(step.value); + if (result !== undefined) return result; + } +}; + + +/***/ }), + +/***/ 2652: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var bind = __webpack_require__(6080); +var call = __webpack_require__(9565); +var anObject = __webpack_require__(8551); +var tryToString = __webpack_require__(6823); +var isArrayIteratorMethod = __webpack_require__(4209); +var lengthOfArrayLike = __webpack_require__(6198); +var isPrototypeOf = __webpack_require__(1625); +var getIterator = __webpack_require__(81); +var getIteratorMethod = __webpack_require__(851); +var iteratorClose = __webpack_require__(9539); + +var $TypeError = TypeError; + +var Result = function (stopped, result) { + this.stopped = stopped; + this.result = result; +}; + +var ResultPrototype = Result.prototype; + +module.exports = function (iterable, unboundFunction, options) { + var that = options && options.that; + var AS_ENTRIES = !!(options && options.AS_ENTRIES); + var IS_RECORD = !!(options && options.IS_RECORD); + var IS_ITERATOR = !!(options && options.IS_ITERATOR); + var INTERRUPTED = !!(options && options.INTERRUPTED); + var fn = bind(unboundFunction, that); + var iterator, iterFn, index, length, result, next, step; + + var stop = function (condition) { + if (iterator) iteratorClose(iterator, 'normal', condition); + return new Result(true, condition); + }; + + var callFn = function (value) { + if (AS_ENTRIES) { + anObject(value); + return INTERRUPTED ? fn(value[0], value[1], stop) : fn(value[0], value[1]); + } return INTERRUPTED ? fn(value, stop) : fn(value); + }; + + if (IS_RECORD) { + iterator = iterable.iterator; + } else if (IS_ITERATOR) { + iterator = iterable; + } else { + iterFn = getIteratorMethod(iterable); + if (!iterFn) throw new $TypeError(tryToString(iterable) + ' is not iterable'); + // optimisation for array iterators + if (isArrayIteratorMethod(iterFn)) { + for (index = 0, length = lengthOfArrayLike(iterable); length > index; index++) { + result = callFn(iterable[index]); + if (result && isPrototypeOf(ResultPrototype, result)) return result; + } return new Result(false); + } + iterator = getIterator(iterable, iterFn); + } + + next = IS_RECORD ? iterable.next : iterator.next; + while (!(step = call(next, iterator)).done) { + try { + result = callFn(step.value); + } catch (error) { + iteratorClose(iterator, 'throw', error); + } + if (typeof result == 'object' && result && isPrototypeOf(ResultPrototype, result)) return result; + } return new Result(false); +}; + + +/***/ }), + +/***/ 9539: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var anObject = __webpack_require__(8551); +var getMethod = __webpack_require__(5966); + +module.exports = function (iterator, kind, value) { + var innerResult, innerError; + anObject(iterator); + try { + innerResult = getMethod(iterator, 'return'); + if (!innerResult) { + if (kind === 'throw') throw value; + return value; + } + innerResult = call(innerResult, iterator); + } catch (error) { + innerError = true; + innerResult = error; + } + if (kind === 'throw') throw value; + if (innerError) throw innerResult; + anObject(innerResult); + return value; +}; + + +/***/ }), + +/***/ 9462: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var create = __webpack_require__(2360); +var createNonEnumerableProperty = __webpack_require__(6699); +var defineBuiltIns = __webpack_require__(6279); +var wellKnownSymbol = __webpack_require__(8227); +var InternalStateModule = __webpack_require__(1181); +var getMethod = __webpack_require__(5966); +var IteratorPrototype = (__webpack_require__(7657).IteratorPrototype); +var createIterResultObject = __webpack_require__(2529); +var iteratorClose = __webpack_require__(9539); + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var ITERATOR_HELPER = 'IteratorHelper'; +var WRAP_FOR_VALID_ITERATOR = 'WrapForValidIterator'; +var setInternalState = InternalStateModule.set; + +var createIteratorProxyPrototype = function (IS_ITERATOR) { + var getInternalState = InternalStateModule.getterFor(IS_ITERATOR ? WRAP_FOR_VALID_ITERATOR : ITERATOR_HELPER); + + return defineBuiltIns(create(IteratorPrototype), { + next: function next() { + var state = getInternalState(this); + // for simplification: + // for `%WrapForValidIteratorPrototype%.next` our `nextHandler` returns `IterResultObject` + // for `%IteratorHelperPrototype%.next` - just a value + if (IS_ITERATOR) return state.nextHandler(); + try { + var result = state.done ? undefined : state.nextHandler(); + return createIterResultObject(result, state.done); + } catch (error) { + state.done = true; + throw error; + } + }, + 'return': function () { + var state = getInternalState(this); + var iterator = state.iterator; + state.done = true; + if (IS_ITERATOR) { + var returnMethod = getMethod(iterator, 'return'); + return returnMethod ? call(returnMethod, iterator) : createIterResultObject(undefined, true); + } + if (state.inner) try { + iteratorClose(state.inner.iterator, 'normal'); + } catch (error) { + return iteratorClose(iterator, 'throw', error); + } + iteratorClose(iterator, 'normal'); + return createIterResultObject(undefined, true); + } + }); +}; + +var WrapForValidIteratorPrototype = createIteratorProxyPrototype(true); +var IteratorHelperPrototype = createIteratorProxyPrototype(false); + +createNonEnumerableProperty(IteratorHelperPrototype, TO_STRING_TAG, 'Iterator Helper'); + +module.exports = function (nextHandler, IS_ITERATOR) { + var IteratorProxy = function Iterator(record, state) { + if (state) { + state.iterator = record.iterator; + state.next = record.next; + } else state = record; + state.type = IS_ITERATOR ? WRAP_FOR_VALID_ITERATOR : ITERATOR_HELPER; + state.nextHandler = nextHandler; + state.counter = 0; + state.done = false; + setInternalState(this, state); + }; + + IteratorProxy.prototype = IS_ITERATOR ? WrapForValidIteratorPrototype : IteratorHelperPrototype; + + return IteratorProxy; +}; + + +/***/ }), + +/***/ 713: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var createIteratorProxy = __webpack_require__(9462); +var callWithSafeIterationClosing = __webpack_require__(6319); + +var IteratorProxy = createIteratorProxy(function () { + var iterator = this.iterator; + var result = anObject(call(this.next, iterator)); + var done = this.done = !!result.done; + if (!done) return callWithSafeIterationClosing(iterator, this.mapper, [result.value, this.counter++], true); +}); + +// `Iterator.prototype.map` method +// https://github.com/tc39/proposal-iterator-helpers +module.exports = function map(mapper) { + anObject(this); + aCallable(mapper); + return new IteratorProxy(getIteratorDirect(this), { + mapper: mapper + }); +}; + + +/***/ }), + +/***/ 7657: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var create = __webpack_require__(2360); +var getPrototypeOf = __webpack_require__(2787); +var defineBuiltIn = __webpack_require__(6840); +var wellKnownSymbol = __webpack_require__(8227); +var IS_PURE = __webpack_require__(6395); + +var ITERATOR = wellKnownSymbol('iterator'); +var BUGGY_SAFARI_ITERATORS = false; + +// `%IteratorPrototype%` object +// https://tc39.es/ecma262/#sec-%iteratorprototype%-object +var IteratorPrototype, PrototypeOfArrayIteratorPrototype, arrayIterator; + +/* eslint-disable es/no-array-prototype-keys -- safe */ +if ([].keys) { + arrayIterator = [].keys(); + // Safari 8 has buggy iterators w/o `next` + if (!('next' in arrayIterator)) BUGGY_SAFARI_ITERATORS = true; + else { + PrototypeOfArrayIteratorPrototype = getPrototypeOf(getPrototypeOf(arrayIterator)); + if (PrototypeOfArrayIteratorPrototype !== Object.prototype) IteratorPrototype = PrototypeOfArrayIteratorPrototype; + } +} + +var NEW_ITERATOR_PROTOTYPE = !isObject(IteratorPrototype) || fails(function () { + var test = {}; + // FF44- legacy iterators case + return IteratorPrototype[ITERATOR].call(test) !== test; +}); + +if (NEW_ITERATOR_PROTOTYPE) IteratorPrototype = {}; +else if (IS_PURE) IteratorPrototype = create(IteratorPrototype); + +// `%IteratorPrototype%[@@iterator]()` method +// https://tc39.es/ecma262/#sec-%iteratorprototype%-@@iterator +if (!isCallable(IteratorPrototype[ITERATOR])) { + defineBuiltIn(IteratorPrototype, ITERATOR, function () { + return this; + }); +} + +module.exports = { + IteratorPrototype: IteratorPrototype, + BUGGY_SAFARI_ITERATORS: BUGGY_SAFARI_ITERATORS +}; + + +/***/ }), + +/***/ 6269: +/***/ ((module) => { + + +module.exports = {}; + + +/***/ }), + +/***/ 6198: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toLength = __webpack_require__(8014); + +// `LengthOfArrayLike` abstract operation +// https://tc39.es/ecma262/#sec-lengthofarraylike +module.exports = function (obj) { + return toLength(obj.length); +}; + + +/***/ }), + +/***/ 283: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var fails = __webpack_require__(9039); +var isCallable = __webpack_require__(4901); +var hasOwn = __webpack_require__(9297); +var DESCRIPTORS = __webpack_require__(3724); +var CONFIGURABLE_FUNCTION_NAME = (__webpack_require__(350).CONFIGURABLE); +var inspectSource = __webpack_require__(3706); +var InternalStateModule = __webpack_require__(1181); + +var enforceInternalState = InternalStateModule.enforce; +var getInternalState = InternalStateModule.get; +var $String = String; +// eslint-disable-next-line es/no-object-defineproperty -- safe +var defineProperty = Object.defineProperty; +var stringSlice = uncurryThis(''.slice); +var replace = uncurryThis(''.replace); +var join = uncurryThis([].join); + +var CONFIGURABLE_LENGTH = DESCRIPTORS && !fails(function () { + return defineProperty(function () { /* empty */ }, 'length', { value: 8 }).length !== 8; +}); + +var TEMPLATE = String(String).split('String'); + +var makeBuiltIn = module.exports = function (value, name, options) { + if (stringSlice($String(name), 0, 7) === 'Symbol(') { + name = '[' + replace($String(name), /^Symbol\(([^)]*)\).*$/, '$1') + ']'; + } + if (options && options.getter) name = 'get ' + name; + if (options && options.setter) name = 'set ' + name; + if (!hasOwn(value, 'name') || (CONFIGURABLE_FUNCTION_NAME && value.name !== name)) { + if (DESCRIPTORS) defineProperty(value, 'name', { value: name, configurable: true }); + else value.name = name; + } + if (CONFIGURABLE_LENGTH && options && hasOwn(options, 'arity') && value.length !== options.arity) { + defineProperty(value, 'length', { value: options.arity }); + } + try { + if (options && hasOwn(options, 'constructor') && options.constructor) { + if (DESCRIPTORS) defineProperty(value, 'prototype', { writable: false }); + // in V8 ~ Chrome 53, prototypes of some methods, like `Array.prototype.values`, are non-writable + } else if (value.prototype) value.prototype = undefined; + } catch (error) { /* empty */ } + var state = enforceInternalState(value); + if (!hasOwn(state, 'source')) { + state.source = join(TEMPLATE, typeof name == 'string' ? name : ''); + } return value; +}; + +// add fake Function#toString for correct work wrapped methods / constructors with methods like LoDash isNative +// eslint-disable-next-line no-extend-native -- required +Function.prototype.toString = makeBuiltIn(function toString() { + return isCallable(this) && getInternalState(this).source || inspectSource(this); +}, 'toString'); + + +/***/ }), + +/***/ 741: +/***/ ((module) => { + + +var ceil = Math.ceil; +var floor = Math.floor; + +// `Math.trunc` method +// https://tc39.es/ecma262/#sec-math.trunc +// eslint-disable-next-line es/no-math-trunc -- safe +module.exports = Math.trunc || function trunc(x) { + var n = +x; + return (n > 0 ? floor : ceil)(n); +}; + + +/***/ }), + +/***/ 6043: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aCallable = __webpack_require__(9306); + +var $TypeError = TypeError; + +var PromiseCapability = function (C) { + var resolve, reject; + this.promise = new C(function ($$resolve, $$reject) { + if (resolve !== undefined || reject !== undefined) throw new $TypeError('Bad Promise constructor'); + resolve = $$resolve; + reject = $$reject; + }); + this.resolve = aCallable(resolve); + this.reject = aCallable(reject); +}; + +// `NewPromiseCapability` abstract operation +// https://tc39.es/ecma262/#sec-newpromisecapability +module.exports.f = function (C) { + return new PromiseCapability(C); +}; + + +/***/ }), + +/***/ 2603: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toString = __webpack_require__(655); + +module.exports = function (argument, $default) { + return argument === undefined ? arguments.length < 2 ? '' : $default : toString(argument); +}; + + +/***/ }), + +/***/ 4149: +/***/ ((module) => { + + +var $RangeError = RangeError; + +module.exports = function (it) { + // eslint-disable-next-line no-self-compare -- NaN check + if (it === it) return it; + throw new $RangeError('NaN is not allowed'); +}; + + +/***/ }), + +/***/ 2360: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* global ActiveXObject -- old IE, WSH */ +var anObject = __webpack_require__(8551); +var definePropertiesModule = __webpack_require__(6801); +var enumBugKeys = __webpack_require__(8727); +var hiddenKeys = __webpack_require__(421); +var html = __webpack_require__(397); +var documentCreateElement = __webpack_require__(4055); +var sharedKey = __webpack_require__(6119); + +var GT = '>'; +var LT = '<'; +var PROTOTYPE = 'prototype'; +var SCRIPT = 'script'; +var IE_PROTO = sharedKey('IE_PROTO'); + +var EmptyConstructor = function () { /* empty */ }; + +var scriptTag = function (content) { + return LT + SCRIPT + GT + content + LT + '/' + SCRIPT + GT; +}; + +// Create object with fake `null` prototype: use ActiveX Object with cleared prototype +var NullProtoObjectViaActiveX = function (activeXDocument) { + activeXDocument.write(scriptTag('')); + activeXDocument.close(); + var temp = activeXDocument.parentWindow.Object; + activeXDocument = null; // avoid memory leak + return temp; +}; + +// Create object with fake `null` prototype: use iframe Object with cleared prototype +var NullProtoObjectViaIFrame = function () { + // Thrash, waste and sodomy: IE GC bug + var iframe = documentCreateElement('iframe'); + var JS = 'java' + SCRIPT + ':'; + var iframeDocument; + iframe.style.display = 'none'; + html.appendChild(iframe); + // https://github.com/zloirock/core-js/issues/475 + iframe.src = String(JS); + iframeDocument = iframe.contentWindow.document; + iframeDocument.open(); + iframeDocument.write(scriptTag('document.F=Object')); + iframeDocument.close(); + return iframeDocument.F; +}; + +// Check for document.domain and active x support +// No need to use active x approach when document.domain is not set +// see https://github.com/es-shims/es5-shim/issues/150 +// variation of https://github.com/kitcambridge/es5-shim/commit/4f738ac066346 +// avoid IE GC bug +var activeXDocument; +var NullProtoObject = function () { + try { + activeXDocument = new ActiveXObject('htmlfile'); + } catch (error) { /* ignore */ } + NullProtoObject = typeof document != 'undefined' + ? document.domain && activeXDocument + ? NullProtoObjectViaActiveX(activeXDocument) // old IE + : NullProtoObjectViaIFrame() + : NullProtoObjectViaActiveX(activeXDocument); // WSH + var length = enumBugKeys.length; + while (length--) delete NullProtoObject[PROTOTYPE][enumBugKeys[length]]; + return NullProtoObject(); +}; + +hiddenKeys[IE_PROTO] = true; + +// `Object.create` method +// https://tc39.es/ecma262/#sec-object.create +// eslint-disable-next-line es/no-object-create -- safe +module.exports = Object.create || function create(O, Properties) { + var result; + if (O !== null) { + EmptyConstructor[PROTOTYPE] = anObject(O); + result = new EmptyConstructor(); + EmptyConstructor[PROTOTYPE] = null; + // add "__proto__" for Object.getPrototypeOf polyfill + result[IE_PROTO] = O; + } else result = NullProtoObject(); + return Properties === undefined ? result : definePropertiesModule.f(result, Properties); +}; + + +/***/ }), + +/***/ 6801: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var V8_PROTOTYPE_DEFINE_BUG = __webpack_require__(8686); +var definePropertyModule = __webpack_require__(4913); +var anObject = __webpack_require__(8551); +var toIndexedObject = __webpack_require__(5397); +var objectKeys = __webpack_require__(1072); + +// `Object.defineProperties` method +// https://tc39.es/ecma262/#sec-object.defineproperties +// eslint-disable-next-line es/no-object-defineproperties -- safe +exports.f = DESCRIPTORS && !V8_PROTOTYPE_DEFINE_BUG ? Object.defineProperties : function defineProperties(O, Properties) { + anObject(O); + var props = toIndexedObject(Properties); + var keys = objectKeys(Properties); + var length = keys.length; + var index = 0; + var key; + while (length > index) definePropertyModule.f(O, key = keys[index++], props[key]); + return O; +}; + + +/***/ }), + +/***/ 4913: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var IE8_DOM_DEFINE = __webpack_require__(5917); +var V8_PROTOTYPE_DEFINE_BUG = __webpack_require__(8686); +var anObject = __webpack_require__(8551); +var toPropertyKey = __webpack_require__(6969); + +var $TypeError = TypeError; +// eslint-disable-next-line es/no-object-defineproperty -- safe +var $defineProperty = Object.defineProperty; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var $getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; +var ENUMERABLE = 'enumerable'; +var CONFIGURABLE = 'configurable'; +var WRITABLE = 'writable'; + +// `Object.defineProperty` method +// https://tc39.es/ecma262/#sec-object.defineproperty +exports.f = DESCRIPTORS ? V8_PROTOTYPE_DEFINE_BUG ? function defineProperty(O, P, Attributes) { + anObject(O); + P = toPropertyKey(P); + anObject(Attributes); + if (typeof O === 'function' && P === 'prototype' && 'value' in Attributes && WRITABLE in Attributes && !Attributes[WRITABLE]) { + var current = $getOwnPropertyDescriptor(O, P); + if (current && current[WRITABLE]) { + O[P] = Attributes.value; + Attributes = { + configurable: CONFIGURABLE in Attributes ? Attributes[CONFIGURABLE] : current[CONFIGURABLE], + enumerable: ENUMERABLE in Attributes ? Attributes[ENUMERABLE] : current[ENUMERABLE], + writable: false + }; + } + } return $defineProperty(O, P, Attributes); +} : $defineProperty : function defineProperty(O, P, Attributes) { + anObject(O); + P = toPropertyKey(P); + anObject(Attributes); + if (IE8_DOM_DEFINE) try { + return $defineProperty(O, P, Attributes); + } catch (error) { /* empty */ } + if ('get' in Attributes || 'set' in Attributes) throw new $TypeError('Accessors not supported'); + if ('value' in Attributes) O[P] = Attributes.value; + return O; +}; + + +/***/ }), + +/***/ 7347: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var call = __webpack_require__(9565); +var propertyIsEnumerableModule = __webpack_require__(8773); +var createPropertyDescriptor = __webpack_require__(6980); +var toIndexedObject = __webpack_require__(5397); +var toPropertyKey = __webpack_require__(6969); +var hasOwn = __webpack_require__(9297); +var IE8_DOM_DEFINE = __webpack_require__(5917); + +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var $getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; + +// `Object.getOwnPropertyDescriptor` method +// https://tc39.es/ecma262/#sec-object.getownpropertydescriptor +exports.f = DESCRIPTORS ? $getOwnPropertyDescriptor : function getOwnPropertyDescriptor(O, P) { + O = toIndexedObject(O); + P = toPropertyKey(P); + if (IE8_DOM_DEFINE) try { + return $getOwnPropertyDescriptor(O, P); + } catch (error) { /* empty */ } + if (hasOwn(O, P)) return createPropertyDescriptor(!call(propertyIsEnumerableModule.f, O, P), O[P]); +}; + + +/***/ }), + +/***/ 8480: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var internalObjectKeys = __webpack_require__(1828); +var enumBugKeys = __webpack_require__(8727); + +var hiddenKeys = enumBugKeys.concat('length', 'prototype'); + +// `Object.getOwnPropertyNames` method +// https://tc39.es/ecma262/#sec-object.getownpropertynames +// eslint-disable-next-line es/no-object-getownpropertynames -- safe +exports.f = Object.getOwnPropertyNames || function getOwnPropertyNames(O) { + return internalObjectKeys(O, hiddenKeys); +}; + + +/***/ }), + +/***/ 3717: +/***/ ((__unused_webpack_module, exports) => { + + +// eslint-disable-next-line es/no-object-getownpropertysymbols -- safe +exports.f = Object.getOwnPropertySymbols; + + +/***/ }), + +/***/ 2787: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var hasOwn = __webpack_require__(9297); +var isCallable = __webpack_require__(4901); +var toObject = __webpack_require__(8981); +var sharedKey = __webpack_require__(6119); +var CORRECT_PROTOTYPE_GETTER = __webpack_require__(2211); + +var IE_PROTO = sharedKey('IE_PROTO'); +var $Object = Object; +var ObjectPrototype = $Object.prototype; + +// `Object.getPrototypeOf` method +// https://tc39.es/ecma262/#sec-object.getprototypeof +// eslint-disable-next-line es/no-object-getprototypeof -- safe +module.exports = CORRECT_PROTOTYPE_GETTER ? $Object.getPrototypeOf : function (O) { + var object = toObject(O); + if (hasOwn(object, IE_PROTO)) return object[IE_PROTO]; + var constructor = object.constructor; + if (isCallable(constructor) && object instanceof constructor) { + return constructor.prototype; + } return object instanceof $Object ? ObjectPrototype : null; +}; + + +/***/ }), + +/***/ 1625: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +module.exports = uncurryThis({}.isPrototypeOf); + + +/***/ }), + +/***/ 1828: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var hasOwn = __webpack_require__(9297); +var toIndexedObject = __webpack_require__(5397); +var indexOf = (__webpack_require__(9617).indexOf); +var hiddenKeys = __webpack_require__(421); + +var push = uncurryThis([].push); + +module.exports = function (object, names) { + var O = toIndexedObject(object); + var i = 0; + var result = []; + var key; + for (key in O) !hasOwn(hiddenKeys, key) && hasOwn(O, key) && push(result, key); + // Don't enum bug & hidden keys + while (names.length > i) if (hasOwn(O, key = names[i++])) { + ~indexOf(result, key) || push(result, key); + } + return result; +}; + + +/***/ }), + +/***/ 1072: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var internalObjectKeys = __webpack_require__(1828); +var enumBugKeys = __webpack_require__(8727); + +// `Object.keys` method +// https://tc39.es/ecma262/#sec-object.keys +// eslint-disable-next-line es/no-object-keys -- safe +module.exports = Object.keys || function keys(O) { + return internalObjectKeys(O, enumBugKeys); +}; + + +/***/ }), + +/***/ 8773: +/***/ ((__unused_webpack_module, exports) => { + + +var $propertyIsEnumerable = {}.propertyIsEnumerable; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; + +// Nashorn ~ JDK8 bug +var NASHORN_BUG = getOwnPropertyDescriptor && !$propertyIsEnumerable.call({ 1: 2 }, 1); + +// `Object.prototype.propertyIsEnumerable` method implementation +// https://tc39.es/ecma262/#sec-object.prototype.propertyisenumerable +exports.f = NASHORN_BUG ? function propertyIsEnumerable(V) { + var descriptor = getOwnPropertyDescriptor(this, V); + return !!descriptor && descriptor.enumerable; +} : $propertyIsEnumerable; + + +/***/ }), + +/***/ 2967: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* eslint-disable no-proto -- safe */ +var uncurryThisAccessor = __webpack_require__(6706); +var isObject = __webpack_require__(34); +var requireObjectCoercible = __webpack_require__(7750); +var aPossiblePrototype = __webpack_require__(3506); + +// `Object.setPrototypeOf` method +// https://tc39.es/ecma262/#sec-object.setprototypeof +// Works with __proto__ only. Old v8 can't work with null proto objects. +// eslint-disable-next-line es/no-object-setprototypeof -- safe +module.exports = Object.setPrototypeOf || ('__proto__' in {} ? function () { + var CORRECT_SETTER = false; + var test = {}; + var setter; + try { + setter = uncurryThisAccessor(Object.prototype, '__proto__', 'set'); + setter(test, []); + CORRECT_SETTER = test instanceof Array; + } catch (error) { /* empty */ } + return function setPrototypeOf(O, proto) { + requireObjectCoercible(O); + aPossiblePrototype(proto); + if (!isObject(O)) return O; + if (CORRECT_SETTER) setter(O, proto); + else O.__proto__ = proto; + return O; + }; +}() : undefined); + + +/***/ }), + +/***/ 4270: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); + +var $TypeError = TypeError; + +// `OrdinaryToPrimitive` abstract operation +// https://tc39.es/ecma262/#sec-ordinarytoprimitive +module.exports = function (input, pref) { + var fn, val; + if (pref === 'string' && isCallable(fn = input.toString) && !isObject(val = call(fn, input))) return val; + if (isCallable(fn = input.valueOf) && !isObject(val = call(fn, input))) return val; + if (pref !== 'string' && isCallable(fn = input.toString) && !isObject(val = call(fn, input))) return val; + throw new $TypeError("Can't convert object to primitive value"); +}; + + +/***/ }), + +/***/ 5031: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); +var uncurryThis = __webpack_require__(9504); +var getOwnPropertyNamesModule = __webpack_require__(8480); +var getOwnPropertySymbolsModule = __webpack_require__(3717); +var anObject = __webpack_require__(8551); + +var concat = uncurryThis([].concat); + +// all object keys, includes non-enumerable and symbols +module.exports = getBuiltIn('Reflect', 'ownKeys') || function ownKeys(it) { + var keys = getOwnPropertyNamesModule.f(anObject(it)); + var getOwnPropertySymbols = getOwnPropertySymbolsModule.f; + return getOwnPropertySymbols ? concat(keys, getOwnPropertySymbols(it)) : keys; +}; + + +/***/ }), + +/***/ 8235: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var hasOwn = __webpack_require__(9297); + +var $SyntaxError = SyntaxError; +var $parseInt = parseInt; +var fromCharCode = String.fromCharCode; +var at = uncurryThis(''.charAt); +var slice = uncurryThis(''.slice); +var exec = uncurryThis(/./.exec); + +var codePoints = { + '\\"': '"', + '\\\\': '\\', + '\\/': '/', + '\\b': '\b', + '\\f': '\f', + '\\n': '\n', + '\\r': '\r', + '\\t': '\t' +}; + +var IS_4_HEX_DIGITS = /^[\da-f]{4}$/i; +// eslint-disable-next-line regexp/no-control-character -- safe +var IS_C0_CONTROL_CODE = /^[\u0000-\u001F]$/; + +module.exports = function (source, i) { + var unterminated = true; + var value = ''; + while (i < source.length) { + var chr = at(source, i); + if (chr === '\\') { + var twoChars = slice(source, i, i + 2); + if (hasOwn(codePoints, twoChars)) { + value += codePoints[twoChars]; + i += 2; + } else if (twoChars === '\\u') { + i += 2; + var fourHexDigits = slice(source, i, i + 4); + if (!exec(IS_4_HEX_DIGITS, fourHexDigits)) throw new $SyntaxError('Bad Unicode escape at: ' + i); + value += fromCharCode($parseInt(fourHexDigits, 16)); + i += 4; + } else throw new $SyntaxError('Unknown escape sequence: "' + twoChars + '"'); + } else if (chr === '"') { + unterminated = false; + i++; + break; + } else { + if (exec(IS_C0_CONTROL_CODE, chr)) throw new $SyntaxError('Bad control character in string literal at: ' + i); + value += chr; + i++; + } + } + if (unterminated) throw new $SyntaxError('Unterminated string at: ' + i); + return { value: value, end: i }; +}; + + +/***/ }), + +/***/ 7750: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isNullOrUndefined = __webpack_require__(4117); + +var $TypeError = TypeError; + +// `RequireObjectCoercible` abstract operation +// https://tc39.es/ecma262/#sec-requireobjectcoercible +module.exports = function (it) { + if (isNullOrUndefined(it)) throw new $TypeError("Can't call method on " + it); + return it; +}; + + +/***/ }), + +/***/ 9286: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var SetHelpers = __webpack_require__(4402); +var iterate = __webpack_require__(8469); + +var Set = SetHelpers.Set; +var add = SetHelpers.add; + +module.exports = function (set) { + var result = new Set(); + iterate(set, function (it) { + add(result, it); + }); + return result; +}; + + +/***/ }), + +/***/ 3440: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var SetHelpers = __webpack_require__(4402); +var clone = __webpack_require__(9286); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSet = __webpack_require__(8469); +var iterateSimple = __webpack_require__(507); + +var has = SetHelpers.has; +var remove = SetHelpers.remove; + +// `Set.prototype.difference` method +// https://github.com/tc39/proposal-set-methods +module.exports = function difference(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + var result = clone(O); + if (size(O) <= otherRec.size) iterateSet(O, function (e) { + if (otherRec.includes(e)) remove(result, e); + }); + else iterateSimple(otherRec.getIterator(), function (e) { + if (has(O, e)) remove(result, e); + }); + return result; +}; + + +/***/ }), + +/***/ 4402: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +// eslint-disable-next-line es/no-set -- safe +var SetPrototype = Set.prototype; + +module.exports = { + // eslint-disable-next-line es/no-set -- safe + Set: Set, + add: uncurryThis(SetPrototype.add), + has: uncurryThis(SetPrototype.has), + remove: uncurryThis(SetPrototype['delete']), + proto: SetPrototype +}; + + +/***/ }), + +/***/ 8750: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var SetHelpers = __webpack_require__(4402); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSet = __webpack_require__(8469); +var iterateSimple = __webpack_require__(507); + +var Set = SetHelpers.Set; +var add = SetHelpers.add; +var has = SetHelpers.has; + +// `Set.prototype.intersection` method +// https://github.com/tc39/proposal-set-methods +module.exports = function intersection(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + var result = new Set(); + + if (size(O) > otherRec.size) { + iterateSimple(otherRec.getIterator(), function (e) { + if (has(O, e)) add(result, e); + }); + } else { + iterateSet(O, function (e) { + if (otherRec.includes(e)) add(result, e); + }); + } + + return result; +}; + + +/***/ }), + +/***/ 4449: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var has = (__webpack_require__(4402).has); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSet = __webpack_require__(8469); +var iterateSimple = __webpack_require__(507); +var iteratorClose = __webpack_require__(9539); + +// `Set.prototype.isDisjointFrom` method +// https://tc39.github.io/proposal-set-methods/#Set.prototype.isDisjointFrom +module.exports = function isDisjointFrom(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + if (size(O) <= otherRec.size) return iterateSet(O, function (e) { + if (otherRec.includes(e)) return false; + }, true) !== false; + var iterator = otherRec.getIterator(); + return iterateSimple(iterator, function (e) { + if (has(O, e)) return iteratorClose(iterator, 'normal', false); + }) !== false; +}; + + +/***/ }), + +/***/ 3838: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var size = __webpack_require__(5170); +var iterate = __webpack_require__(8469); +var getSetRecord = __webpack_require__(3789); + +// `Set.prototype.isSubsetOf` method +// https://tc39.github.io/proposal-set-methods/#Set.prototype.isSubsetOf +module.exports = function isSubsetOf(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + if (size(O) > otherRec.size) return false; + return iterate(O, function (e) { + if (!otherRec.includes(e)) return false; + }, true) !== false; +}; + + +/***/ }), + +/***/ 8527: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var has = (__webpack_require__(4402).has); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSimple = __webpack_require__(507); +var iteratorClose = __webpack_require__(9539); + +// `Set.prototype.isSupersetOf` method +// https://tc39.github.io/proposal-set-methods/#Set.prototype.isSupersetOf +module.exports = function isSupersetOf(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + if (size(O) < otherRec.size) return false; + var iterator = otherRec.getIterator(); + return iterateSimple(iterator, function (e) { + if (!has(O, e)) return iteratorClose(iterator, 'normal', false); + }) !== false; +}; + + +/***/ }), + +/***/ 8469: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var iterateSimple = __webpack_require__(507); +var SetHelpers = __webpack_require__(4402); + +var Set = SetHelpers.Set; +var SetPrototype = SetHelpers.proto; +var forEach = uncurryThis(SetPrototype.forEach); +var keys = uncurryThis(SetPrototype.keys); +var next = keys(new Set()).next; + +module.exports = function (set, fn, interruptible) { + return interruptible ? iterateSimple({ iterator: keys(set), next: next }, fn) : forEach(set, fn); +}; + + +/***/ }), + +/***/ 4916: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); + +var createSetLike = function (size) { + return { + size: size, + has: function () { + return false; + }, + keys: function () { + return { + next: function () { + return { done: true }; + } + }; + } + }; +}; + +module.exports = function (name) { + var Set = getBuiltIn('Set'); + try { + new Set()[name](createSetLike(0)); + try { + // late spec change, early WebKit ~ Safari 17.0 beta implementation does not pass it + // https://github.com/tc39/proposal-set-methods/pull/88 + new Set()[name](createSetLike(-1)); + return false; + } catch (error2) { + return true; + } + } catch (error) { + return false; + } +}; + + +/***/ }), + +/***/ 5170: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThisAccessor = __webpack_require__(6706); +var SetHelpers = __webpack_require__(4402); + +module.exports = uncurryThisAccessor(SetHelpers.proto, 'size', 'get') || function (set) { + return set.size; +}; + + +/***/ }), + +/***/ 3650: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var SetHelpers = __webpack_require__(4402); +var clone = __webpack_require__(9286); +var getSetRecord = __webpack_require__(3789); +var iterateSimple = __webpack_require__(507); + +var add = SetHelpers.add; +var has = SetHelpers.has; +var remove = SetHelpers.remove; + +// `Set.prototype.symmetricDifference` method +// https://github.com/tc39/proposal-set-methods +module.exports = function symmetricDifference(other) { + var O = aSet(this); + var keysIter = getSetRecord(other).getIterator(); + var result = clone(O); + iterateSimple(keysIter, function (e) { + if (has(O, e)) remove(result, e); + else add(result, e); + }); + return result; +}; + + +/***/ }), + +/***/ 4204: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var add = (__webpack_require__(4402).add); +var clone = __webpack_require__(9286); +var getSetRecord = __webpack_require__(3789); +var iterateSimple = __webpack_require__(507); + +// `Set.prototype.union` method +// https://github.com/tc39/proposal-set-methods +module.exports = function union(other) { + var O = aSet(this); + var keysIter = getSetRecord(other).getIterator(); + var result = clone(O); + iterateSimple(keysIter, function (it) { + add(result, it); + }); + return result; +}; + + +/***/ }), + +/***/ 6119: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var shared = __webpack_require__(5745); +var uid = __webpack_require__(3392); + +var keys = shared('keys'); + +module.exports = function (key) { + return keys[key] || (keys[key] = uid(key)); +}; + + +/***/ }), + +/***/ 7629: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var IS_PURE = __webpack_require__(6395); +var globalThis = __webpack_require__(4475); +var defineGlobalProperty = __webpack_require__(9433); + +var SHARED = '__core-js_shared__'; +var store = module.exports = globalThis[SHARED] || defineGlobalProperty(SHARED, {}); + +(store.versions || (store.versions = [])).push({ + version: '3.37.1', + mode: IS_PURE ? 'pure' : 'global', + copyright: '© 2014-2024 Denis Pushkarev (zloirock.ru)', + license: 'https://github.com/zloirock/core-js/blob/v3.37.1/LICENSE', + source: 'https://github.com/zloirock/core-js' +}); + + +/***/ }), + +/***/ 5745: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var store = __webpack_require__(7629); + +module.exports = function (key, value) { + return store[key] || (store[key] = value || {}); +}; + + +/***/ }), + +/***/ 1548: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var fails = __webpack_require__(9039); +var V8 = __webpack_require__(7388); +var IS_BROWSER = __webpack_require__(7290); +var IS_DENO = __webpack_require__(516); +var IS_NODE = __webpack_require__(9088); + +var structuredClone = global.structuredClone; + +module.exports = !!structuredClone && !fails(function () { + // prevent V8 ArrayBufferDetaching protector cell invalidation and performance degradation + // https://github.com/zloirock/core-js/issues/679 + if ((IS_DENO && V8 > 92) || (IS_NODE && V8 > 94) || (IS_BROWSER && V8 > 97)) return false; + var buffer = new ArrayBuffer(8); + var clone = structuredClone(buffer, { transfer: [buffer] }); + return buffer.byteLength !== 0 || clone.byteLength !== 8; +}); + + +/***/ }), + +/***/ 4495: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* eslint-disable es/no-symbol -- required for testing */ +var V8_VERSION = __webpack_require__(7388); +var fails = __webpack_require__(9039); +var global = __webpack_require__(4475); + +var $String = global.String; + +// eslint-disable-next-line es/no-object-getownpropertysymbols -- required for testing +module.exports = !!Object.getOwnPropertySymbols && !fails(function () { + var symbol = Symbol('symbol detection'); + // Chrome 38 Symbol has incorrect toString conversion + // `get-own-property-symbols` polyfill symbols converted to object are not Symbol instances + // nb: Do not call `String` directly to avoid this being optimized out to `symbol+''` which will, + // of course, fail. + return !$String(symbol) || !(Object(symbol) instanceof Symbol) || + // Chrome 38-40 symbols are not inherited from DOM collections prototypes to instances + !Symbol.sham && V8_VERSION && V8_VERSION < 41; +}); + + +/***/ }), + +/***/ 5610: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); + +var max = Math.max; +var min = Math.min; + +// Helper for a popular repeating case of the spec: +// Let integer be ? ToInteger(index). +// If integer < 0, let result be max((length + integer), 0); else let result be min(integer, length). +module.exports = function (index, length) { + var integer = toIntegerOrInfinity(index); + return integer < 0 ? max(integer + length, 0) : min(integer, length); +}; + + +/***/ }), + +/***/ 5854: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toPrimitive = __webpack_require__(2777); + +var $TypeError = TypeError; + +// `ToBigInt` abstract operation +// https://tc39.es/ecma262/#sec-tobigint +module.exports = function (argument) { + var prim = toPrimitive(argument, 'number'); + if (typeof prim == 'number') throw new $TypeError("Can't convert number to bigint"); + // eslint-disable-next-line es/no-bigint -- safe + return BigInt(prim); +}; + + +/***/ }), + +/***/ 7696: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); +var toLength = __webpack_require__(8014); + +var $RangeError = RangeError; + +// `ToIndex` abstract operation +// https://tc39.es/ecma262/#sec-toindex +module.exports = function (it) { + if (it === undefined) return 0; + var number = toIntegerOrInfinity(it); + var length = toLength(number); + if (number !== length) throw new $RangeError('Wrong length or index'); + return length; +}; + + +/***/ }), + +/***/ 5397: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +// toObject with fallback for non-array-like ES3 strings +var IndexedObject = __webpack_require__(7055); +var requireObjectCoercible = __webpack_require__(7750); + +module.exports = function (it) { + return IndexedObject(requireObjectCoercible(it)); +}; + + +/***/ }), + +/***/ 1291: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var trunc = __webpack_require__(741); + +// `ToIntegerOrInfinity` abstract operation +// https://tc39.es/ecma262/#sec-tointegerorinfinity +module.exports = function (argument) { + var number = +argument; + // eslint-disable-next-line no-self-compare -- NaN check + return number !== number || number === 0 ? 0 : trunc(number); +}; + + +/***/ }), + +/***/ 8014: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); + +var min = Math.min; + +// `ToLength` abstract operation +// https://tc39.es/ecma262/#sec-tolength +module.exports = function (argument) { + var len = toIntegerOrInfinity(argument); + return len > 0 ? min(len, 0x1FFFFFFFFFFFFF) : 0; // 2 ** 53 - 1 == 9007199254740991 +}; + + +/***/ }), + +/***/ 8981: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var requireObjectCoercible = __webpack_require__(7750); + +var $Object = Object; + +// `ToObject` abstract operation +// https://tc39.es/ecma262/#sec-toobject +module.exports = function (argument) { + return $Object(requireObjectCoercible(argument)); +}; + + +/***/ }), + +/***/ 9590: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); + +var $RangeError = RangeError; + +module.exports = function (it) { + var result = toIntegerOrInfinity(it); + if (result < 0) throw new $RangeError("The argument can't be less than 0"); + return result; +}; + + +/***/ }), + +/***/ 2777: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var isObject = __webpack_require__(34); +var isSymbol = __webpack_require__(757); +var getMethod = __webpack_require__(5966); +var ordinaryToPrimitive = __webpack_require__(4270); +var wellKnownSymbol = __webpack_require__(8227); + +var $TypeError = TypeError; +var TO_PRIMITIVE = wellKnownSymbol('toPrimitive'); + +// `ToPrimitive` abstract operation +// https://tc39.es/ecma262/#sec-toprimitive +module.exports = function (input, pref) { + if (!isObject(input) || isSymbol(input)) return input; + var exoticToPrim = getMethod(input, TO_PRIMITIVE); + var result; + if (exoticToPrim) { + if (pref === undefined) pref = 'default'; + result = call(exoticToPrim, input, pref); + if (!isObject(result) || isSymbol(result)) return result; + throw new $TypeError("Can't convert object to primitive value"); + } + if (pref === undefined) pref = 'number'; + return ordinaryToPrimitive(input, pref); +}; + + +/***/ }), + +/***/ 6969: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toPrimitive = __webpack_require__(2777); +var isSymbol = __webpack_require__(757); + +// `ToPropertyKey` abstract operation +// https://tc39.es/ecma262/#sec-topropertykey +module.exports = function (argument) { + var key = toPrimitive(argument, 'string'); + return isSymbol(key) ? key : key + ''; +}; + + +/***/ }), + +/***/ 2140: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var wellKnownSymbol = __webpack_require__(8227); + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var test = {}; + +test[TO_STRING_TAG] = 'z'; + +module.exports = String(test) === '[object z]'; + + +/***/ }), + +/***/ 655: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(6955); + +var $String = String; + +module.exports = function (argument) { + if (classof(argument) === 'Symbol') throw new TypeError('Cannot convert a Symbol value to a string'); + return $String(argument); +}; + + +/***/ }), + +/***/ 9714: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var IS_NODE = __webpack_require__(9088); + +module.exports = function (name) { + try { + // eslint-disable-next-line no-new-func -- safe + if (IS_NODE) return Function('return require("' + name + '")')(); + } catch (error) { /* empty */ } +}; + + +/***/ }), + +/***/ 6823: +/***/ ((module) => { + + +var $String = String; + +module.exports = function (argument) { + try { + return $String(argument); + } catch (error) { + return 'Object'; + } +}; + + +/***/ }), + +/***/ 3392: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +var id = 0; +var postfix = Math.random(); +var toString = uncurryThis(1.0.toString); + +module.exports = function (key) { + return 'Symbol(' + (key === undefined ? '' : key) + ')_' + toString(++id + postfix, 36); +}; + + +/***/ }), + +/***/ 7040: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* eslint-disable es/no-symbol -- required for testing */ +var NATIVE_SYMBOL = __webpack_require__(4495); + +module.exports = NATIVE_SYMBOL + && !Symbol.sham + && typeof Symbol.iterator == 'symbol'; + + +/***/ }), + +/***/ 8686: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var fails = __webpack_require__(9039); + +// V8 ~ Chrome 36- +// https://bugs.chromium.org/p/v8/issues/detail?id=3334 +module.exports = DESCRIPTORS && fails(function () { + // eslint-disable-next-line es/no-object-defineproperty -- required for testing + return Object.defineProperty(function () { /* empty */ }, 'prototype', { + value: 42, + writable: false + }).prototype !== 42; +}); + + +/***/ }), + +/***/ 2812: +/***/ ((module) => { + + +var $TypeError = TypeError; + +module.exports = function (passed, required) { + if (passed < required) throw new $TypeError('Not enough arguments'); + return passed; +}; + + +/***/ }), + +/***/ 8622: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var isCallable = __webpack_require__(4901); + +var WeakMap = global.WeakMap; + +module.exports = isCallable(WeakMap) && /native code/.test(String(WeakMap)); + + +/***/ }), + +/***/ 8227: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var shared = __webpack_require__(5745); +var hasOwn = __webpack_require__(9297); +var uid = __webpack_require__(3392); +var NATIVE_SYMBOL = __webpack_require__(4495); +var USE_SYMBOL_AS_UID = __webpack_require__(7040); + +var Symbol = global.Symbol; +var WellKnownSymbolsStore = shared('wks'); +var createWellKnownSymbol = USE_SYMBOL_AS_UID ? Symbol['for'] || Symbol : Symbol && Symbol.withoutSetter || uid; + +module.exports = function (name) { + if (!hasOwn(WellKnownSymbolsStore, name)) { + WellKnownSymbolsStore[name] = NATIVE_SYMBOL && hasOwn(Symbol, name) + ? Symbol[name] + : createWellKnownSymbol('Symbol.' + name); + } return WellKnownSymbolsStore[name]; +}; + + +/***/ }), + +/***/ 6573: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var defineBuiltInAccessor = __webpack_require__(2106); +var isDetached = __webpack_require__(3238); + +var ArrayBufferPrototype = ArrayBuffer.prototype; + +if (DESCRIPTORS && !('detached' in ArrayBufferPrototype)) { + defineBuiltInAccessor(ArrayBufferPrototype, 'detached', { + configurable: true, + get: function detached() { + return isDetached(this); + } + }); +} + + +/***/ }), + +/***/ 7936: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var $transfer = __webpack_require__(5636); + +// `ArrayBuffer.prototype.transferToFixedLength` method +// https://tc39.es/proposal-arraybuffer-transfer/#sec-arraybuffer.prototype.transfertofixedlength +if ($transfer) $({ target: 'ArrayBuffer', proto: true }, { + transferToFixedLength: function transferToFixedLength() { + return $transfer(this, arguments.length ? arguments[0] : undefined, false); + } +}); + + +/***/ }), + +/***/ 8100: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var $transfer = __webpack_require__(5636); + +// `ArrayBuffer.prototype.transfer` method +// https://tc39.es/proposal-arraybuffer-transfer/#sec-arraybuffer.prototype.transfer +if ($transfer) $({ target: 'ArrayBuffer', proto: true }, { + transfer: function transfer() { + return $transfer(this, arguments.length ? arguments[0] : undefined, true); + } +}); + + +/***/ }), + +/***/ 4114: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var toObject = __webpack_require__(8981); +var lengthOfArrayLike = __webpack_require__(6198); +var setArrayLength = __webpack_require__(4527); +var doesNotExceedSafeInteger = __webpack_require__(6837); +var fails = __webpack_require__(9039); + +var INCORRECT_TO_LENGTH = fails(function () { + return [].push.call({ length: 0x100000000 }, 1) !== 4294967297; +}); + +// V8 <= 121 and Safari <= 15.4; FF < 23 throws InternalError +// https://bugs.chromium.org/p/v8/issues/detail?id=12681 +var properErrorOnNonWritableLength = function () { + try { + // eslint-disable-next-line es/no-object-defineproperty -- safe + Object.defineProperty([], 'length', { writable: false }).push(); + } catch (error) { + return error instanceof TypeError; + } +}; + +var FORCED = INCORRECT_TO_LENGTH || !properErrorOnNonWritableLength(); + +// `Array.prototype.push` method +// https://tc39.es/ecma262/#sec-array.prototype.push +$({ target: 'Array', proto: true, arity: 1, forced: FORCED }, { + // eslint-disable-next-line no-unused-vars -- required for `.length` + push: function push(item) { + var O = toObject(this); + var len = lengthOfArrayLike(O); + var argCount = arguments.length; + doesNotExceedSafeInteger(len + argCount); + for (var i = 0; i < argCount; i++) { + O[len] = arguments[i]; + len++; + } + setArrayLength(O, len); + return len; + } +}); + + +/***/ }), + +/***/ 4628: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var newPromiseCapabilityModule = __webpack_require__(6043); + +// `Promise.withResolvers` method +// https://github.com/tc39/proposal-promise-with-resolvers +$({ target: 'Promise', stat: true }, { + withResolvers: function withResolvers() { + var promiseCapability = newPromiseCapabilityModule.f(this); + return { + promise: promiseCapability.promise, + resolve: promiseCapability.resolve, + reject: promiseCapability.reject + }; + } +}); + + +/***/ }), + +/***/ 7642: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var difference = __webpack_require__(3440); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.difference` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('difference') }, { + difference: difference +}); + + +/***/ }), + +/***/ 8004: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var fails = __webpack_require__(9039); +var intersection = __webpack_require__(8750); +var setMethodAcceptSetLike = __webpack_require__(4916); + +var INCORRECT = !setMethodAcceptSetLike('intersection') || fails(function () { + // eslint-disable-next-line es/no-array-from, es/no-set -- testing + return String(Array.from(new Set([1, 2, 3]).intersection(new Set([3, 2])))) !== '3,2'; +}); + +// `Set.prototype.intersection` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: INCORRECT }, { + intersection: intersection +}); + + +/***/ }), + +/***/ 3853: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var isDisjointFrom = __webpack_require__(4449); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.isDisjointFrom` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('isDisjointFrom') }, { + isDisjointFrom: isDisjointFrom +}); + + +/***/ }), + +/***/ 5876: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var isSubsetOf = __webpack_require__(3838); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.isSubsetOf` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('isSubsetOf') }, { + isSubsetOf: isSubsetOf +}); + + +/***/ }), + +/***/ 2475: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var isSupersetOf = __webpack_require__(8527); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.isSupersetOf` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('isSupersetOf') }, { + isSupersetOf: isSupersetOf +}); + + +/***/ }), + +/***/ 5024: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var symmetricDifference = __webpack_require__(3650); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.symmetricDifference` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('symmetricDifference') }, { + symmetricDifference: symmetricDifference +}); + + +/***/ }), + +/***/ 1698: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var union = __webpack_require__(4204); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.union` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('union') }, { + union: union +}); + + +/***/ }), + +/***/ 7467: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var arrayToReversed = __webpack_require__(7628); +var ArrayBufferViewCore = __webpack_require__(4644); + +var aTypedArray = ArrayBufferViewCore.aTypedArray; +var exportTypedArrayMethod = ArrayBufferViewCore.exportTypedArrayMethod; +var getTypedArrayConstructor = ArrayBufferViewCore.getTypedArrayConstructor; + +// `%TypedArray%.prototype.toReversed` method +// https://tc39.es/ecma262/#sec-%typedarray%.prototype.toreversed +exportTypedArrayMethod('toReversed', function toReversed() { + return arrayToReversed(aTypedArray(this), getTypedArrayConstructor(this)); +}); + + +/***/ }), + +/***/ 4732: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var ArrayBufferViewCore = __webpack_require__(4644); +var uncurryThis = __webpack_require__(9504); +var aCallable = __webpack_require__(9306); +var arrayFromConstructorAndList = __webpack_require__(5370); + +var aTypedArray = ArrayBufferViewCore.aTypedArray; +var getTypedArrayConstructor = ArrayBufferViewCore.getTypedArrayConstructor; +var exportTypedArrayMethod = ArrayBufferViewCore.exportTypedArrayMethod; +var sort = uncurryThis(ArrayBufferViewCore.TypedArrayPrototype.sort); + +// `%TypedArray%.prototype.toSorted` method +// https://tc39.es/ecma262/#sec-%typedarray%.prototype.tosorted +exportTypedArrayMethod('toSorted', function toSorted(compareFn) { + if (compareFn !== undefined) aCallable(compareFn); + var O = aTypedArray(this); + var A = arrayFromConstructorAndList(getTypedArrayConstructor(O), O); + return sort(A, compareFn); +}); + + +/***/ }), + +/***/ 9577: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var arrayWith = __webpack_require__(9928); +var ArrayBufferViewCore = __webpack_require__(4644); +var isBigIntArray = __webpack_require__(1108); +var toIntegerOrInfinity = __webpack_require__(1291); +var toBigInt = __webpack_require__(5854); + +var aTypedArray = ArrayBufferViewCore.aTypedArray; +var getTypedArrayConstructor = ArrayBufferViewCore.getTypedArrayConstructor; +var exportTypedArrayMethod = ArrayBufferViewCore.exportTypedArrayMethod; + +var PROPER_ORDER = !!function () { + try { + // eslint-disable-next-line no-throw-literal, es/no-typed-arrays, es/no-array-prototype-with -- required for testing + new Int8Array(1)['with'](2, { valueOf: function () { throw 8; } }); + } catch (error) { + // some early implementations, like WebKit, does not follow the final semantic + // https://github.com/tc39/proposal-change-array-by-copy/pull/86 + return error === 8; + } +}(); + +// `%TypedArray%.prototype.with` method +// https://tc39.es/ecma262/#sec-%typedarray%.prototype.with +exportTypedArrayMethod('with', { 'with': function (index, value) { + var O = aTypedArray(this); + var relativeIndex = toIntegerOrInfinity(index); + var actualValue = isBigIntArray(O) ? toBigInt(value) : +value; + return arrayWith(O, getTypedArrayConstructor(O), relativeIndex, actualValue); +} }['with'], !PROPER_ORDER); + + +/***/ }), + +/***/ 8992: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var global = __webpack_require__(4475); +var anInstance = __webpack_require__(679); +var anObject = __webpack_require__(8551); +var isCallable = __webpack_require__(4901); +var getPrototypeOf = __webpack_require__(2787); +var defineBuiltInAccessor = __webpack_require__(2106); +var createProperty = __webpack_require__(4659); +var fails = __webpack_require__(9039); +var hasOwn = __webpack_require__(9297); +var wellKnownSymbol = __webpack_require__(8227); +var IteratorPrototype = (__webpack_require__(7657).IteratorPrototype); +var DESCRIPTORS = __webpack_require__(3724); +var IS_PURE = __webpack_require__(6395); + +var CONSTRUCTOR = 'constructor'; +var ITERATOR = 'Iterator'; +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); + +var $TypeError = TypeError; +var NativeIterator = global[ITERATOR]; + +// FF56- have non-standard global helper `Iterator` +var FORCED = IS_PURE + || !isCallable(NativeIterator) + || NativeIterator.prototype !== IteratorPrototype + // FF44- non-standard `Iterator` passes previous tests + || !fails(function () { NativeIterator({}); }); + +var IteratorConstructor = function Iterator() { + anInstance(this, IteratorPrototype); + if (getPrototypeOf(this) === IteratorPrototype) throw new $TypeError('Abstract class Iterator not directly constructable'); +}; + +var defineIteratorPrototypeAccessor = function (key, value) { + if (DESCRIPTORS) { + defineBuiltInAccessor(IteratorPrototype, key, { + configurable: true, + get: function () { + return value; + }, + set: function (replacement) { + anObject(this); + if (this === IteratorPrototype) throw new $TypeError("You can't redefine this property"); + if (hasOwn(this, key)) this[key] = replacement; + else createProperty(this, key, replacement); + } + }); + } else IteratorPrototype[key] = value; +}; + +if (!hasOwn(IteratorPrototype, TO_STRING_TAG)) defineIteratorPrototypeAccessor(TO_STRING_TAG, ITERATOR); + +if (FORCED || !hasOwn(IteratorPrototype, CONSTRUCTOR) || IteratorPrototype[CONSTRUCTOR] === Object) { + defineIteratorPrototypeAccessor(CONSTRUCTOR, IteratorConstructor); +} + +IteratorConstructor.prototype = IteratorPrototype; + +// `Iterator` constructor +// https://github.com/tc39/proposal-iterator-helpers +$({ global: true, constructor: true, forced: FORCED }, { + Iterator: IteratorConstructor +}); + + +/***/ }), + +/***/ 4743: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var call = __webpack_require__(9565); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var notANaN = __webpack_require__(4149); +var toPositiveInteger = __webpack_require__(9590); +var createIteratorProxy = __webpack_require__(9462); +var IS_PURE = __webpack_require__(6395); + +var IteratorProxy = createIteratorProxy(function () { + var iterator = this.iterator; + var next = this.next; + var result, done; + while (this.remaining) { + this.remaining--; + result = anObject(call(next, iterator)); + done = this.done = !!result.done; + if (done) return; + } + result = anObject(call(next, iterator)); + done = this.done = !!result.done; + if (!done) return result.value; +}); + +// `Iterator.prototype.drop` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true, forced: IS_PURE }, { + drop: function drop(limit) { + anObject(this); + var remaining = toPositiveInteger(notANaN(+limit)); + return new IteratorProxy(getIteratorDirect(this), { + remaining: remaining + }); + } +}); + + +/***/ }), + +/***/ 3215: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +// `Iterator.prototype.every` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + every: function every(predicate) { + anObject(this); + aCallable(predicate); + var record = getIteratorDirect(this); + var counter = 0; + return !iterate(record, function (value, stop) { + if (!predicate(value, counter++)) return stop(); + }, { IS_RECORD: true, INTERRUPTED: true }).stopped; + } +}); + + +/***/ }), + +/***/ 4520: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var createIteratorProxy = __webpack_require__(9462); +var callWithSafeIterationClosing = __webpack_require__(6319); +var IS_PURE = __webpack_require__(6395); + +var IteratorProxy = createIteratorProxy(function () { + var iterator = this.iterator; + var predicate = this.predicate; + var next = this.next; + var result, done, value; + while (true) { + result = anObject(call(next, iterator)); + done = this.done = !!result.done; + if (done) return; + value = result.value; + if (callWithSafeIterationClosing(iterator, predicate, [value, this.counter++], true)) return value; + } +}); + +// `Iterator.prototype.filter` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true, forced: IS_PURE }, { + filter: function filter(predicate) { + anObject(this); + aCallable(predicate); + return new IteratorProxy(getIteratorDirect(this), { + predicate: predicate + }); + } +}); + + +/***/ }), + +/***/ 670: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var getIteratorFlattenable = __webpack_require__(8646); +var createIteratorProxy = __webpack_require__(9462); +var iteratorClose = __webpack_require__(9539); +var IS_PURE = __webpack_require__(6395); + +var IteratorProxy = createIteratorProxy(function () { + var iterator = this.iterator; + var mapper = this.mapper; + var result, inner; + + while (true) { + if (inner = this.inner) try { + result = anObject(call(inner.next, inner.iterator)); + if (!result.done) return result.value; + this.inner = null; + } catch (error) { iteratorClose(iterator, 'throw', error); } + + result = anObject(call(this.next, iterator)); + + if (this.done = !!result.done) return; + + try { + this.inner = getIteratorFlattenable(mapper(result.value, this.counter++), false); + } catch (error) { iteratorClose(iterator, 'throw', error); } + } +}); + +// `Iterator.prototype.flatMap` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true, forced: IS_PURE }, { + flatMap: function flatMap(mapper) { + anObject(this); + aCallable(mapper); + return new IteratorProxy(getIteratorDirect(this), { + mapper: mapper, + inner: null + }); + } +}); + + +/***/ }), + +/***/ 1454: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var map = __webpack_require__(713); +var IS_PURE = __webpack_require__(6395); + +// `Iterator.prototype.map` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true, forced: IS_PURE }, { + map: map +}); + + +/***/ }), + +/***/ 7550: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +// `Iterator.prototype.some` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + some: function some(predicate) { + anObject(this); + aCallable(predicate); + var record = getIteratorDirect(this); + var counter = 0; + return iterate(record, function (value, stop) { + if (predicate(value, counter++)) return stop(); + }, { IS_RECORD: true, INTERRUPTED: true }).stopped; + } +}); + + +/***/ }), + +/***/ 8335: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var DESCRIPTORS = __webpack_require__(3724); +var global = __webpack_require__(4475); +var getBuiltIn = __webpack_require__(7751); +var uncurryThis = __webpack_require__(9504); +var call = __webpack_require__(9565); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var isArray = __webpack_require__(4376); +var hasOwn = __webpack_require__(9297); +var toString = __webpack_require__(655); +var lengthOfArrayLike = __webpack_require__(6198); +var createProperty = __webpack_require__(4659); +var fails = __webpack_require__(9039); +var parseJSONString = __webpack_require__(8235); +var NATIVE_SYMBOL = __webpack_require__(4495); + +var JSON = global.JSON; +var Number = global.Number; +var SyntaxError = global.SyntaxError; +var nativeParse = JSON && JSON.parse; +var enumerableOwnProperties = getBuiltIn('Object', 'keys'); +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; +var at = uncurryThis(''.charAt); +var slice = uncurryThis(''.slice); +var exec = uncurryThis(/./.exec); +var push = uncurryThis([].push); + +var IS_DIGIT = /^\d$/; +var IS_NON_ZERO_DIGIT = /^[1-9]$/; +var IS_NUMBER_START = /^(?:-|\d)$/; +var IS_WHITESPACE = /^[\t\n\r ]$/; + +var PRIMITIVE = 0; +var OBJECT = 1; + +var $parse = function (source, reviver) { + source = toString(source); + var context = new Context(source, 0, ''); + var root = context.parse(); + var value = root.value; + var endIndex = context.skip(IS_WHITESPACE, root.end); + if (endIndex < source.length) { + throw new SyntaxError('Unexpected extra character: "' + at(source, endIndex) + '" after the parsed data at: ' + endIndex); + } + return isCallable(reviver) ? internalize({ '': value }, '', reviver, root) : value; +}; + +var internalize = function (holder, name, reviver, node) { + var val = holder[name]; + var unmodified = node && val === node.value; + var context = unmodified && typeof node.source == 'string' ? { source: node.source } : {}; + var elementRecordsLen, keys, len, i, P; + if (isObject(val)) { + var nodeIsArray = isArray(val); + var nodes = unmodified ? node.nodes : nodeIsArray ? [] : {}; + if (nodeIsArray) { + elementRecordsLen = nodes.length; + len = lengthOfArrayLike(val); + for (i = 0; i < len; i++) { + internalizeProperty(val, i, internalize(val, '' + i, reviver, i < elementRecordsLen ? nodes[i] : undefined)); + } + } else { + keys = enumerableOwnProperties(val); + len = lengthOfArrayLike(keys); + for (i = 0; i < len; i++) { + P = keys[i]; + internalizeProperty(val, P, internalize(val, P, reviver, hasOwn(nodes, P) ? nodes[P] : undefined)); + } + } + } + return call(reviver, holder, name, val, context); +}; + +var internalizeProperty = function (object, key, value) { + if (DESCRIPTORS) { + var descriptor = getOwnPropertyDescriptor(object, key); + if (descriptor && !descriptor.configurable) return; + } + if (value === undefined) delete object[key]; + else createProperty(object, key, value); +}; + +var Node = function (value, end, source, nodes) { + this.value = value; + this.end = end; + this.source = source; + this.nodes = nodes; +}; + +var Context = function (source, index) { + this.source = source; + this.index = index; +}; + +// https://www.json.org/json-en.html +Context.prototype = { + fork: function (nextIndex) { + return new Context(this.source, nextIndex); + }, + parse: function () { + var source = this.source; + var i = this.skip(IS_WHITESPACE, this.index); + var fork = this.fork(i); + var chr = at(source, i); + if (exec(IS_NUMBER_START, chr)) return fork.number(); + switch (chr) { + case '{': + return fork.object(); + case '[': + return fork.array(); + case '"': + return fork.string(); + case 't': + return fork.keyword(true); + case 'f': + return fork.keyword(false); + case 'n': + return fork.keyword(null); + } throw new SyntaxError('Unexpected character: "' + chr + '" at: ' + i); + }, + node: function (type, value, start, end, nodes) { + return new Node(value, end, type ? null : slice(this.source, start, end), nodes); + }, + object: function () { + var source = this.source; + var i = this.index + 1; + var expectKeypair = false; + var object = {}; + var nodes = {}; + while (i < source.length) { + i = this.until(['"', '}'], i); + if (at(source, i) === '}' && !expectKeypair) { + i++; + break; + } + // Parsing the key + var result = this.fork(i).string(); + var key = result.value; + i = result.end; + i = this.until([':'], i) + 1; + // Parsing value + i = this.skip(IS_WHITESPACE, i); + result = this.fork(i).parse(); + createProperty(nodes, key, result); + createProperty(object, key, result.value); + i = this.until([',', '}'], result.end); + var chr = at(source, i); + if (chr === ',') { + expectKeypair = true; + i++; + } else if (chr === '}') { + i++; + break; + } + } + return this.node(OBJECT, object, this.index, i, nodes); + }, + array: function () { + var source = this.source; + var i = this.index + 1; + var expectElement = false; + var array = []; + var nodes = []; + while (i < source.length) { + i = this.skip(IS_WHITESPACE, i); + if (at(source, i) === ']' && !expectElement) { + i++; + break; + } + var result = this.fork(i).parse(); + push(nodes, result); + push(array, result.value); + i = this.until([',', ']'], result.end); + if (at(source, i) === ',') { + expectElement = true; + i++; + } else if (at(source, i) === ']') { + i++; + break; + } + } + return this.node(OBJECT, array, this.index, i, nodes); + }, + string: function () { + var index = this.index; + var parsed = parseJSONString(this.source, this.index + 1); + return this.node(PRIMITIVE, parsed.value, index, parsed.end); + }, + number: function () { + var source = this.source; + var startIndex = this.index; + var i = startIndex; + if (at(source, i) === '-') i++; + if (at(source, i) === '0') i++; + else if (exec(IS_NON_ZERO_DIGIT, at(source, i))) i = this.skip(IS_DIGIT, ++i); + else throw new SyntaxError('Failed to parse number at: ' + i); + if (at(source, i) === '.') i = this.skip(IS_DIGIT, ++i); + if (at(source, i) === 'e' || at(source, i) === 'E') { + i++; + if (at(source, i) === '+' || at(source, i) === '-') i++; + var exponentStartIndex = i; + i = this.skip(IS_DIGIT, i); + if (exponentStartIndex === i) throw new SyntaxError("Failed to parse number's exponent value at: " + i); + } + return this.node(PRIMITIVE, Number(slice(source, startIndex, i)), startIndex, i); + }, + keyword: function (value) { + var keyword = '' + value; + var index = this.index; + var endIndex = index + keyword.length; + if (slice(this.source, index, endIndex) !== keyword) throw new SyntaxError('Failed to parse value at: ' + index); + return this.node(PRIMITIVE, value, index, endIndex); + }, + skip: function (regex, i) { + var source = this.source; + for (; i < source.length; i++) if (!exec(regex, at(source, i))) break; + return i; + }, + until: function (array, i) { + i = this.skip(IS_WHITESPACE, i); + var chr = at(this.source, i); + for (var j = 0; j < array.length; j++) if (array[j] === chr) return i; + throw new SyntaxError('Unexpected character: "' + chr + '" at: ' + i); + } +}; + +var NO_SOURCE_SUPPORT = fails(function () { + var unsafeInt = '9007199254740993'; + var source; + nativeParse(unsafeInt, function (key, value, context) { + source = context.source; + }); + return source !== unsafeInt; +}); + +var PROPER_BASE_PARSE = NATIVE_SYMBOL && !fails(function () { + // Safari 9 bug + return 1 / nativeParse('-0 \t') !== -Infinity; +}); + +// `JSON.parse` method +// https://tc39.es/ecma262/#sec-json.parse +// https://github.com/tc39/proposal-json-parse-with-source +$({ target: 'JSON', stat: true, forced: NO_SOURCE_SUPPORT }, { + parse: function parse(text, reviver) { + return PROPER_BASE_PARSE && !isCallable(reviver) ? nativeParse(text) : $parse(text, reviver); + } +}); + + +/***/ }), + +/***/ 3375: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(7642); + + +/***/ }), + +/***/ 9225: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(8004); + + +/***/ }), + +/***/ 3972: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(3853); + + +/***/ }), + +/***/ 9209: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(5876); + + +/***/ }), + +/***/ 5714: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(2475); + + +/***/ }), + +/***/ 7561: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(5024); + + +/***/ }), + +/***/ 6197: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(1698); + + +/***/ }), + +/***/ 4979: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var global = __webpack_require__(4475); +var getBuiltIn = __webpack_require__(7751); +var createPropertyDescriptor = __webpack_require__(6980); +var defineProperty = (__webpack_require__(4913).f); +var hasOwn = __webpack_require__(9297); +var anInstance = __webpack_require__(679); +var inheritIfRequired = __webpack_require__(3167); +var normalizeStringArgument = __webpack_require__(2603); +var DOMExceptionConstants = __webpack_require__(5002); +var clearErrorStack = __webpack_require__(6193); +var DESCRIPTORS = __webpack_require__(3724); +var IS_PURE = __webpack_require__(6395); + +var DOM_EXCEPTION = 'DOMException'; +var Error = getBuiltIn('Error'); +var NativeDOMException = getBuiltIn(DOM_EXCEPTION); + +var $DOMException = function DOMException() { + anInstance(this, DOMExceptionPrototype); + var argumentsLength = arguments.length; + var message = normalizeStringArgument(argumentsLength < 1 ? undefined : arguments[0]); + var name = normalizeStringArgument(argumentsLength < 2 ? undefined : arguments[1], 'Error'); + var that = new NativeDOMException(message, name); + var error = new Error(message); + error.name = DOM_EXCEPTION; + defineProperty(that, 'stack', createPropertyDescriptor(1, clearErrorStack(error.stack, 1))); + inheritIfRequired(that, this, $DOMException); + return that; +}; + +var DOMExceptionPrototype = $DOMException.prototype = NativeDOMException.prototype; + +var ERROR_HAS_STACK = 'stack' in new Error(DOM_EXCEPTION); +var DOM_EXCEPTION_HAS_STACK = 'stack' in new NativeDOMException(1, 2); + +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var descriptor = NativeDOMException && DESCRIPTORS && Object.getOwnPropertyDescriptor(global, DOM_EXCEPTION); + +// Bun ~ 0.1.1 DOMException have incorrect descriptor and we can't redefine it +// https://github.com/Jarred-Sumner/bun/issues/399 +var BUGGY_DESCRIPTOR = !!descriptor && !(descriptor.writable && descriptor.configurable); + +var FORCED_CONSTRUCTOR = ERROR_HAS_STACK && !BUGGY_DESCRIPTOR && !DOM_EXCEPTION_HAS_STACK; + +// `DOMException` constructor patch for `.stack` where it's required +// https://webidl.spec.whatwg.org/#es-DOMException-specialness +$({ global: true, constructor: true, forced: IS_PURE || FORCED_CONSTRUCTOR }, { // TODO: fix export logic + DOMException: FORCED_CONSTRUCTOR ? $DOMException : NativeDOMException +}); + +var PolyfilledDOMException = getBuiltIn(DOM_EXCEPTION); +var PolyfilledDOMExceptionPrototype = PolyfilledDOMException.prototype; + +if (PolyfilledDOMExceptionPrototype.constructor !== PolyfilledDOMException) { + if (!IS_PURE) { + defineProperty(PolyfilledDOMExceptionPrototype, 'constructor', createPropertyDescriptor(1, PolyfilledDOMException)); + } + + for (var key in DOMExceptionConstants) if (hasOwn(DOMExceptionConstants, key)) { + var constant = DOMExceptionConstants[key]; + var constantName = constant.s; + if (!hasOwn(PolyfilledDOMException, constantName)) { + defineProperty(PolyfilledDOMException, constantName, createPropertyDescriptor(6, constant.c)); + } + } +} + + +/***/ }), + +/***/ 4603: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var defineBuiltIn = __webpack_require__(6840); +var uncurryThis = __webpack_require__(9504); +var toString = __webpack_require__(655); +var validateArgumentsLength = __webpack_require__(2812); + +var $URLSearchParams = URLSearchParams; +var URLSearchParamsPrototype = $URLSearchParams.prototype; +var append = uncurryThis(URLSearchParamsPrototype.append); +var $delete = uncurryThis(URLSearchParamsPrototype['delete']); +var forEach = uncurryThis(URLSearchParamsPrototype.forEach); +var push = uncurryThis([].push); +var params = new $URLSearchParams('a=1&a=2&b=3'); + +params['delete']('a', 1); +// `undefined` case is a Chromium 117 bug +// https://bugs.chromium.org/p/v8/issues/detail?id=14222 +params['delete']('b', undefined); + +if (params + '' !== 'a=2') { + defineBuiltIn(URLSearchParamsPrototype, 'delete', function (name /* , value */) { + var length = arguments.length; + var $value = length < 2 ? undefined : arguments[1]; + if (length && $value === undefined) return $delete(this, name); + var entries = []; + forEach(this, function (v, k) { // also validates `this` + push(entries, { key: k, value: v }); + }); + validateArgumentsLength(length, 1); + var key = toString(name); + var value = toString($value); + var index = 0; + var dindex = 0; + var found = false; + var entriesLength = entries.length; + var entry; + while (index < entriesLength) { + entry = entries[index++]; + if (found || entry.key === key) { + found = true; + $delete(this, entry.key); + } else dindex++; + } + while (dindex < entriesLength) { + entry = entries[dindex++]; + if (!(entry.key === key && entry.value === value)) append(this, entry.key, entry.value); + } + }, { enumerable: true, unsafe: true }); +} + + +/***/ }), + +/***/ 7566: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var defineBuiltIn = __webpack_require__(6840); +var uncurryThis = __webpack_require__(9504); +var toString = __webpack_require__(655); +var validateArgumentsLength = __webpack_require__(2812); + +var $URLSearchParams = URLSearchParams; +var URLSearchParamsPrototype = $URLSearchParams.prototype; +var getAll = uncurryThis(URLSearchParamsPrototype.getAll); +var $has = uncurryThis(URLSearchParamsPrototype.has); +var params = new $URLSearchParams('a=1'); + +// `undefined` case is a Chromium 117 bug +// https://bugs.chromium.org/p/v8/issues/detail?id=14222 +if (params.has('a', 2) || !params.has('a', undefined)) { + defineBuiltIn(URLSearchParamsPrototype, 'has', function has(name /* , value */) { + var length = arguments.length; + var $value = length < 2 ? undefined : arguments[1]; + if (length && $value === undefined) return $has(this, name); + var values = getAll(this, name); // also validates `this` + validateArgumentsLength(length, 1); + var value = toString($value); + var index = 0; + while (index < values.length) { + if (values[index++] === value) return true; + } return false; + }, { enumerable: true, unsafe: true }); +} + + +/***/ }), + +/***/ 8721: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var uncurryThis = __webpack_require__(9504); +var defineBuiltInAccessor = __webpack_require__(2106); + +var URLSearchParamsPrototype = URLSearchParams.prototype; +var forEach = uncurryThis(URLSearchParamsPrototype.forEach); + +// `URLSearchParams.prototype.size` getter +// https://github.com/whatwg/url/pull/734 +if (DESCRIPTORS && !('size' in URLSearchParamsPrototype)) { + defineBuiltInAccessor(URLSearchParamsPrototype, 'size', { + get: function size() { + var count = 0; + forEach(this, function () { count++; }); + return count; + }, + configurable: true, + enumerable: true + }); +} + + +/***/ }) + +/******/ }); +/************************************************************************/ +/******/ // The module cache +/******/ var __webpack_module_cache__ = {}; +/******/ +/******/ // The require function +/******/ function __webpack_require__(moduleId) { +/******/ // Check if module is in cache +/******/ var cachedModule = __webpack_module_cache__[moduleId]; +/******/ if (cachedModule !== undefined) { +/******/ return cachedModule.exports; +/******/ } +/******/ // Create a new module (and put it into the cache) +/******/ var module = __webpack_module_cache__[moduleId] = { +/******/ // no module.id needed +/******/ // no module.loaded needed +/******/ exports: {} +/******/ }; +/******/ +/******/ // Execute the module function +/******/ __webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__); +/******/ +/******/ // Return the exports of the module +/******/ return module.exports; +/******/ } /******/ /************************************************************************/ /******/ /* webpack/runtime/define property getters */ @@ -63,7 +4767,6 @@ __webpack_require__.d(__webpack_exports__, { InvalidPDFException: () => (/* reexport */ InvalidPDFException), MissingPDFException: () => (/* reexport */ MissingPDFException), OPS: () => (/* reexport */ OPS), - Outliner: () => (/* reexport */ Outliner), PDFDataRangeTransport: () => (/* reexport */ PDFDataRangeTransport), PDFDateString: () => (/* reexport */ PDFDateString), PDFWorker: () => (/* reexport */ PDFWorker), @@ -87,14 +4790,42 @@ __webpack_require__.d(__webpack_exports__, { isPdfFile: () => (/* reexport */ isPdfFile), noContextMenu: () => (/* reexport */ noContextMenu), normalizeUnicode: () => (/* reexport */ normalizeUnicode), - renderTextLayer: () => (/* reexport */ renderTextLayer), setLayerDimensions: () => (/* reexport */ setLayerDimensions), shadow: () => (/* reexport */ shadow), - updateTextLayer: () => (/* reexport */ updateTextLayer), version: () => (/* reexport */ version) }); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array.push.js +var es_array_push = __webpack_require__(4114); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array-buffer.detached.js +var es_array_buffer_detached = __webpack_require__(6573); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array-buffer.transfer.js +var es_array_buffer_transfer = __webpack_require__(8100); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array-buffer.transfer-to-fixed-length.js +var es_array_buffer_transfer_to_fixed_length = __webpack_require__(7936); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.typed-array.to-reversed.js +var es_typed_array_to_reversed = __webpack_require__(7467); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.typed-array.to-sorted.js +var es_typed_array_to_sorted = __webpack_require__(4732); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.typed-array.with.js +var es_typed_array_with = __webpack_require__(9577); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.url-search-params.delete.js +var web_url_search_params_delete = __webpack_require__(4603); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.url-search-params.has.js +var web_url_search_params_has = __webpack_require__(7566); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.url-search-params.size.js +var web_url_search_params_size = __webpack_require__(8721); ;// CONCATENATED MODULE: ./src/shared/util.js + + + + + + + + + + const isNodeJS = typeof process === "object" && process + "" === "[object process]" && !process.versions.nw && !(process.versions.electron && process.type && process.type !== "browser"); const IDENTITY_MATRIX = [1, 0, 0, 1, 0, 0]; const FONT_IDENTITY_MATRIX = [0.001, 0, 0, 0.001, 0, 0]; @@ -110,6 +4841,7 @@ const RenderingIntentFlag = { ANNOTATIONS_FORMS: 0x10, ANNOTATIONS_STORAGE: 0x20, ANNOTATIONS_DISABLE: 0x40, + IS_EDITING: 0x80, OPLIST: 0x100 }; const AnnotationMode = { @@ -365,7 +5097,9 @@ const OPS = { paintImageXObjectRepeat: 88, paintImageMaskXObjectRepeat: 89, paintSolidColorImageMask: 90, - constructPath: 91 + constructPath: 91, + setStrokeTransparent: 92, + setFillTransparent: 93 }; const PasswordResponses = { NEED_PASSWORD: 1, @@ -830,6 +5564,38 @@ const FontRenderOps = { TRANSLATE: 8 }; +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.promise.with-resolvers.js +var es_promise_with_resolvers = __webpack_require__(4628); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.map.js +var esnext_iterator_map = __webpack_require__(1454); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.difference.v2.js +var esnext_set_difference_v2 = __webpack_require__(3375); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.intersection.v2.js +var esnext_set_intersection_v2 = __webpack_require__(9225); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.is-disjoint-from.v2.js +var esnext_set_is_disjoint_from_v2 = __webpack_require__(3972); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.is-subset-of.v2.js +var esnext_set_is_subset_of_v2 = __webpack_require__(9209); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.is-superset-of.v2.js +var esnext_set_is_superset_of_v2 = __webpack_require__(5714); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.symmetric-difference.v2.js +var esnext_set_symmetric_difference_v2 = __webpack_require__(7561); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.union.v2.js +var esnext_set_union_v2 = __webpack_require__(6197); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.dom-exception.stack.js +var web_dom_exception_stack = __webpack_require__(4979); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.constructor.js +var esnext_iterator_constructor = __webpack_require__(8992); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.drop.js +var esnext_iterator_drop = __webpack_require__(4743); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.every.js +var esnext_iterator_every = __webpack_require__(3215); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.some.js +var esnext_iterator_some = __webpack_require__(7550); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.json.parse.js +var esnext_json_parse = __webpack_require__(8335); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.filter.js +var esnext_iterator_filter = __webpack_require__(4520); ;// CONCATENATED MODULE: ./src/display/base_factory.js class BaseFilterFactory { @@ -991,6 +5757,19 @@ class BaseSVGFactory { ;// CONCATENATED MODULE: ./src/display/display_utils.js + + + + + + + + + + + + + const SVG_NS = "http://www.w3.org/2000/svg"; class PixelsPerInch { static CSS = 96.0; @@ -998,6 +5777,7 @@ class PixelsPerInch { static PDF_TO_CSS_UNITS = this.CSS / this.PDF; } class DOMFilterFactory extends BaseFilterFactory { + #baseUrl; #_cache; #_defs; #docId; @@ -1061,6 +5841,20 @@ class DOMFilterFactory extends BaseFilterFactory { } return [bufferR.join(","), bufferG.join(","), bufferB.join(",")]; } + #createUrl(id) { + if (this.#baseUrl === undefined) { + this.#baseUrl = ""; + const url = this.#document.URL; + if (url !== this.#document.baseURI) { + if (isDataScheme(url)) { + warn('#createUrl: ignore "data:"-URL for performance reasons.'); + } else { + this.#baseUrl = url.split("#", 1)[0]; + } + } + } + return `url(${this.#baseUrl}#${id})`; + } addFilter(maps) { if (!maps) { return "none"; @@ -1077,7 +5871,7 @@ class DOMFilterFactory extends BaseFilterFactory { return value; } const id = `g_${this.#docId}_transfer_map_${this.#id++}`; - const url = `url(#${id})`; + const url = this.#createUrl(id); this.#cache.set(maps, url); this.#cache.set(key, url); const filter = this.#createFilter(id); @@ -1135,7 +5929,7 @@ class DOMFilterFactory extends BaseFilterFactory { return arr.join(","); }; this.#addTransferMapConversion(getSteps(0, 5), getSteps(1, 5), getSteps(2, 5), filter); - info.url = `url(#${id})`; + info.url = this.#createUrl(id); return info.url; } addAlphaFilter(map) { @@ -1151,7 +5945,7 @@ class DOMFilterFactory extends BaseFilterFactory { return value; } const id = `g_${this.#docId}_alpha_map_${this.#id++}`; - const url = `url(#${id})`; + const url = this.#createUrl(id); this.#cache.set(map, url); this.#cache.set(key, url); const filter = this.#createFilter(id); @@ -1176,7 +5970,7 @@ class DOMFilterFactory extends BaseFilterFactory { return value; } const id = `g_${this.#docId}_luminosity_map_${this.#id++}`; - const url = `url(#${id})`; + const url = this.#createUrl(id); this.#cache.set(map, url); this.#cache.set(key, url); const filter = this.#createFilter(id); @@ -1239,7 +6033,7 @@ class DOMFilterFactory extends BaseFilterFactory { const filter = info.filter = this.#createFilter(id); this.#addGrayConversion(filter); this.#addTransferMapConversion(getSteps(newFgRGB[0], newBgRGB[0], 5), getSteps(newFgRGB[1], newBgRGB[1], 5), getSteps(newFgRGB[2], newBgRGB[2], 5), filter); - info.url = `url(#${id})`; + info.url = this.#createUrl(id); return info.url; } destroy(keepHCM = false) { @@ -1890,6 +6684,26 @@ class HighlightToolbar { ;// CONCATENATED MODULE: ./src/display/editor/tools.js + + + + + + + + + + + + + + + + + + + + function bindEvents(obj, element, names) { for (const name of names) { @@ -2234,6 +7048,7 @@ class AnnotationEditorUIManager { #editorTypes = null; #editorsToRescale = new Set(); #enableHighlightFloatingButton = false; + #enableUpdatedAddImage = false; #filterFactory = null; #focusMainContainerTimeoutId = null; #highlightColors = null; @@ -2333,7 +7148,7 @@ class AnnotationEditorUIManager { checker: arrowChecker }]])); } - constructor(container, viewer, altTextManager, eventBus, pdfDocument, pageColors, highlightColors, enableHighlightFloatingButton, mlManager) { + constructor(container, viewer, altTextManager, eventBus, pdfDocument, pageColors, highlightColors, enableHighlightFloatingButton, enableUpdatedAddImage, mlManager) { this._signal = this.#abortController.signal; this.#container = container; this.#viewer = viewer; @@ -2351,6 +7166,7 @@ class AnnotationEditorUIManager { this.#pageColors = pageColors; this.#highlightColors = highlightColors || null; this.#enableHighlightFloatingButton = enableHighlightFloatingButton; + this.#enableUpdatedAddImage = enableUpdatedAddImage; this.#mlManager = mlManager || null; this.viewParameters = { realScale: PixelsPerInch.PDF_TO_CSS_UNITS, @@ -2390,8 +7206,11 @@ class AnnotationEditorUIManager { async mlGuess(data) { return this.#mlManager?.guess(data) || null; } - get hasMLManager() { - return !!this.#mlManager; + async isMLEnabledFor(name) { + return !!(await this.#mlManager?.isEnabledFor(name)); + } + get useNewAltTextFlow() { + return this.#enableUpdatedAddImage; } get hcmFilter() { return shadow(this, "hcmFilter", this.#pageColors ? this.#filterFactory.addHCMFilter(this.#pageColors.foreground, this.#pageColors.background) : "none"); @@ -2411,6 +7230,23 @@ class AnnotationEditorUIManager { editAltText(editor) { this.#altTextManager?.editAltText(this, editor); } + switchToMode(mode, callback) { + this._eventBus.on("annotationeditormodechanged", callback, { + once: true, + signal: this._signal + }); + this._eventBus.dispatch("showannotationeditorui", { + source: this, + mode + }); + } + setPreference(name, value) { + this._eventBus.dispatch("setpreference", { + source: this, + name, + value + }); + } onPageChanging({ pageNumber }) { @@ -2462,6 +7298,20 @@ class AnnotationEditorUIManager { }) { return anchorNode.nodeType === Node.TEXT_NODE ? anchorNode.parentElement : anchorNode; } + #getLayerForTextLayer(textLayer) { + const { + currentLayer + } = this; + if (currentLayer.hasTextLayer(textLayer)) { + return currentLayer; + } + for (const layer of this.#allLayers.values()) { + if (layer.hasTextLayer(textLayer)) { + return layer; + } + } + return null; + } highlightSelection(methodOfCreation = "") { const selection = document.getSelection(); if (!selection || selection.isCollapsed) { @@ -2481,30 +7331,30 @@ class AnnotationEditorUIManager { return; } selection.empty(); - if (this.#mode === AnnotationEditorType.NONE) { - this._eventBus.dispatch("showannotationeditorui", { - source: this, - mode: AnnotationEditorType.HIGHLIGHT + const layer = this.#getLayerForTextLayer(textLayer); + const isNoneMode = this.#mode === AnnotationEditorType.NONE; + const callback = () => { + layer?.createAndAddNewEditor({ + x: 0, + y: 0 + }, false, { + methodOfCreation, + boxes, + anchorNode, + anchorOffset, + focusNode, + focusOffset, + text }); - this.showAllEditors("highlight", true, true); - } - for (const layer of this.#allLayers.values()) { - if (layer.hasTextLayer(textLayer)) { - layer.createAndAddNewEditor({ - x: 0, - y: 0 - }, false, { - methodOfCreation, - boxes, - anchorNode, - anchorOffset, - focusNode, - focusOffset, - text - }); - break; + if (isNoneMode) { + this.showAllEditors("highlight", true, true); } + }; + if (isNoneMode) { + this.switchToMode(AnnotationEditorType.HIGHLIGHT, callback); + return; } + callback(); } #displayHighlightToolbar() { const selection = document.getSelection(); @@ -2568,11 +7418,14 @@ class AnnotationEditorUIManager { } this.#highlightWhenShiftUp = this.isShiftKeyDown; if (!this.isShiftKeyDown) { + const activeLayer = this.#mode === AnnotationEditorType.HIGHLIGHT ? this.#getLayerForTextLayer(textLayer) : null; + activeLayer?.toggleDrawing(); const signal = this._signal; const pointerup = e => { if (e.type === "pointerup" && e.button !== 0) { return; } + activeLayer?.toggleDrawing(true); window.removeEventListener("pointerup", pointerup); window.removeEventListener("blur", pointerup); if (e.type === "pointerup") { @@ -4824,6 +9677,12 @@ class FakeEditor extends AnnotationEditor { } ;// CONCATENATED MODULE: ./src/shared/murmurhash3.js + + + + + + const SEED = 0xc3d2e1f0; const MASK_HIGH = 0xffff0000; const MASK_LOW = 0xffff; @@ -4920,6 +9779,16 @@ class MurmurHash3_64 { + + + + + + + + + + const SerializableEmpty = Object.freeze({ map: null, hash: "", @@ -4927,6 +9796,7 @@ const SerializableEmpty = Object.freeze({ }); class AnnotationStorage { #modified = false; + #modifiedIds = null; #storage = new Map(); constructor() { this.onSetModified = null; @@ -5078,6 +9948,25 @@ class AnnotationStorage { } return stats; } + resetModifiedIds() { + this.#modifiedIds = null; + } + get modifiedIds() { + if (this.#modifiedIds) { + return this.#modifiedIds; + } + const ids = []; + for (const value of this.#storage.values()) { + if (!(value instanceof AnnotationEditor) || !value.annotationElementId || !value.serialize()) { + continue; + } + ids.push(value.annotationElementId); + } + return this.#modifiedIds = { + ids: new Set(ids), + hash: ids.join(",") + }; + } } class PrintAnnotationStorage extends AnnotationStorage { #serializable; @@ -5103,10 +9992,25 @@ class PrintAnnotationStorage extends AnnotationStorage { get serializable() { return this.#serializable; } + get modifiedIds() { + return shadow(this, "modifiedIds", { + ids: new Set(), + hash: "" + }); + } } ;// CONCATENATED MODULE: ./src/display/font_loader.js + + + + + + + + + class FontLoader { #systemFonts = new Set(); constructor({ @@ -5444,6 +10348,13 @@ class FontFaceObject { ;// CONCATENATED MODULE: ./src/display/node_utils.js + + + + + + + if (isNodeJS) { var packageCapability = Promise.withResolvers(); var packageMap = null; @@ -5453,6 +10364,12 @@ if (isNodeJS) { https = await import( /*webpackIgnore: true*/"https"), url = await import( /*webpackIgnore: true*/"url"); let canvas, path2d; + try { + canvas = await import( /*webpackIgnore: true*/"canvas"); + } catch {} + try { + path2d = await import( /*webpackIgnore: true*/"path2d"); + } catch {} return new Map(Object.entries({ fs, http, @@ -5465,6 +10382,25 @@ if (isNodeJS) { loadPackages().then(map => { packageMap = map; packageCapability.resolve(); + if (!globalThis.DOMMatrix) { + const DOMMatrix = map.get("canvas")?.DOMMatrix; + if (DOMMatrix) { + globalThis.DOMMatrix = DOMMatrix; + } else { + warn("Cannot polyfill `DOMMatrix`, rendering may be broken."); + } + } + if (!globalThis.Path2D) { + const CanvasRenderingContext2D = map.get("canvas")?.CanvasRenderingContext2D; + const applyPath2DToCanvasRenderingContext = map.get("path2d")?.applyPath2DToCanvasRenderingContext; + const Path2D = map.get("path2d")?.Path2D; + if (CanvasRenderingContext2D && applyPath2DToCanvasRenderingContext && Path2D) { + applyPath2DToCanvasRenderingContext(CanvasRenderingContext2D); + globalThis.Path2D = Path2D; + } else { + warn("Cannot polyfill `Path2D`, rendering may be broken."); + } + } }, reason => { warn(`loadPackages: ${reason}`); packageMap = new Map(); @@ -5946,6 +10882,12 @@ class TilingPattern { ;// CONCATENATED MODULE: ./src/shared/image_utils.js + + + + + + function convertToRGBA(params) { switch (params.kind) { case ImageKind.GRAYSCALE_1BPP: @@ -6056,6 +10998,17 @@ function grayToRGBA(src, dest) { + + + + + + + + + + + const MIN_FONT_SIZE = 16; const MAX_FONT_SIZE = 100; const EXECUTION_TIME = 15; @@ -6749,6 +11702,7 @@ class CanvasGraphics { while (this.stateStack.length || this.inSMaskMode) { this.restore(); } + this.current.activeSMask = null; this.ctx.restore(); if (this.transparentCanvas) { this.ctx = this.compositeCtx; @@ -7655,14 +12609,17 @@ class CanvasGraphics { this.current.patternFill = true; } setStrokeRGBColor(r, g, b) { - const color = Util.makeHexColor(r, g, b); - this.ctx.strokeStyle = color; - this.current.strokeColor = color; + this.ctx.strokeStyle = this.current.strokeColor = Util.makeHexColor(r, g, b); + } + setStrokeTransparent() { + this.ctx.strokeStyle = this.current.strokeColor = "transparent"; } setFillRGBColor(r, g, b) { - const color = Util.makeHexColor(r, g, b); - this.ctx.fillStyle = color; - this.current.fillColor = color; + this.ctx.fillStyle = this.current.fillColor = Util.makeHexColor(r, g, b); + this.current.patternFill = false; + } + setFillTransparent() { + this.ctx.fillStyle = this.current.fillColor = "transparent"; this.current.patternFill = false; } _getPattern(objId, matrix = null) { @@ -8252,6 +13209,7 @@ class GlobalWorkerOptions { ;// CONCATENATED MODULE: ./src/shared/message_handler.js + const CallbackKind = { UNKNOWN: 0, DATA: 1, @@ -8934,6 +13892,16 @@ class OptionalContentConfig { ;// CONCATENATED MODULE: ./src/display/transport_stream.js + + + + + + + + + + class PDFDataTransportStream { constructor(pdfDataRangeTransport, { disableRange = false, @@ -9201,6 +14169,8 @@ class PDFDataTransportStreamRangeReader { ;// CONCATENATED MODULE: ./src/display/content_disposition.js + + function getFilenameFromContentDispositionHeader(contentDisposition) { let needsEncodingFixup = true; let tmp = toParamRegExp("filename\\*", "i").exec(contentDisposition); @@ -9393,6 +14363,14 @@ function validateResponseStatus(status) { ;// CONCATENATED MODULE: ./src/display/fetch_stream.js + + + + + + + + function createFetchOptions(headers, withCredentials, abortController) { return { method: "GET", @@ -9597,6 +14575,8 @@ class PDFFetchStreamRangeReader { ;// CONCATENATED MODULE: ./src/display/network.js + + const OK_RESPONSE = 200; const PARTIAL_CONTENT_RESPONSE = 206; function network_getArrayBuffer(xhr) { @@ -10004,6 +14984,14 @@ class PDFNetworkStreamRangeRequestReader { + + + + + + + + const fileUriRegex = /^file:\/\/\/[a-zA-Z]:\//; function parseUrl(sourceUrl) { const url = NodePackages.get("url"); @@ -10342,6 +15330,15 @@ class PDFNodeStreamFsRangeReader extends BaseRangeReader { ;// CONCATENATED MODULE: ./src/display/text_layer.js + + + + + + + + + const MAX_TEXT_DIVS_TO_RENDER = 100000; const DEFAULT_FONT_SIZE = 30; const DEFAULT_FONT_ASCENT = 0.8; @@ -10657,6 +15654,7 @@ class TextLayer { div.style.opacity = 0; div.style.lineHeight = 1; div.style.fontSize = "1px"; + div.style.position = "absolute"; div.textContent = "X"; document.body.append(div); this.#minFontSize = div.getBoundingClientRect().height; @@ -10709,39 +15707,9 @@ class TextLayer { return ratio; } } -function renderTextLayer() { - deprecated("`renderTextLayer`, please use `TextLayer` instead."); - const { - textContentSource, - container, - viewport, - ...rest - } = arguments[0]; - const restKeys = Object.keys(rest); - if (restKeys.length > 0) { - warn("Ignoring `renderTextLayer` parameters: " + restKeys.join(", ")); - } - const textLayer = new TextLayer({ - textContentSource, - container, - viewport - }); - const { - textDivs, - textContentItemsStr - } = textLayer; - const promise = textLayer.render(); - return { - promise, - textDivs, - textContentItemsStr - }; -} -function updateTextLayer() { - deprecated("`updateTextLayer`, please use `TextLayer` instead."); -} ;// CONCATENATED MODULE: ./src/display/xfa_text.js + class XfaText { static textContent(xfa) { const items = []; @@ -10800,6 +15768,26 @@ class XfaText { + + + + + + + + + + + + + + + + + + + + const DEFAULT_RANGE_CHUNK_SIZE = 65536; const RENDERING_CANCELLED_TIMEOUT = 100; @@ -10887,7 +15875,7 @@ function getDocument(src = {}) { } const docParams = { docId, - apiVersion: "4.4.168", + apiVersion: "4.5.136", data, password, disableAutoFetch, @@ -11280,10 +16268,11 @@ class PDFPageProxy { optionalContentConfigPromise = null, annotationCanvasMap = null, pageColors = null, - printAnnotationStorage = null + printAnnotationStorage = null, + isEditing = false }) { this._stats?.time("Overall"); - const intentArgs = this._transport.getRenderingIntent(intent, annotationMode, printAnnotationStorage); + const intentArgs = this._transport.getRenderingIntent(intent, annotationMode, printAnnotationStorage, isEditing); const { renderingIntent, cacheKey @@ -11376,7 +16365,8 @@ class PDFPageProxy { getOperatorList({ intent = "display", annotationMode = AnnotationMode.ENABLE, - printAnnotationStorage = null + printAnnotationStorage = null, + isEditing = false } = {}) { function operatorListChanged() { if (intentState.operatorList.lastChunk) { @@ -11384,7 +16374,7 @@ class PDFPageProxy { intentState.renderTasks.delete(opListTask); } } - const intentArgs = this._transport.getRenderingIntent(intent, annotationMode, printAnnotationStorage, true); + const intentArgs = this._transport.getRenderingIntent(intent, annotationMode, printAnnotationStorage, isEditing, true); let intentState = this._intentStates.get(intentArgs.cacheKey); if (!intentState) { intentState = Object.create(null); @@ -11542,7 +16532,8 @@ class PDFPageProxy { _pumpOperatorList({ renderingIntent, cacheKey, - annotationStorageSerializable + annotationStorageSerializable, + modifiedIds }) { const { map, @@ -11552,7 +16543,8 @@ class PDFPageProxy { pageIndex: this._pageIndex, intent: renderingIntent, cacheKey, - annotationStorage: map + annotationStorage: map, + modifiedIds }, transfer); const reader = readableStream.getReader(); const intentState = this._intentStates.get(cacheKey); @@ -11675,7 +16667,7 @@ const PDFWorkerUtil = { { if (isNodeJS) { PDFWorkerUtil.isWorkerDisabled = true; - GlobalWorkerOptions.workerSrc ||= "./pdf.worker.mjs"; + GlobalWorkerOptions.workerSrc ||= "./pdf.worker.js"; } PDFWorkerUtil.isSameOrigin = function (baseUrl, otherUrl) { let base; @@ -11924,7 +16916,7 @@ class WorkerTransport { get annotationStorage() { return shadow(this, "annotationStorage", new AnnotationStorage()); } - getRenderingIntent(intent, annotationMode = AnnotationMode.ENABLE, printAnnotationStorage = null, isOpList = false) { + getRenderingIntent(intent, annotationMode = AnnotationMode.ENABLE, printAnnotationStorage = null, isEditing = false, isOpList = false) { let renderingIntent = RenderingIntentFlag.DISPLAY; let annotationStorageSerializable = SerializableEmpty; switch (intent) { @@ -11939,6 +16931,7 @@ class WorkerTransport { default: warn(`getRenderingIntent - invalid intent: ${intent}`); } + const annotationStorage = renderingIntent & RenderingIntentFlag.PRINT && printAnnotationStorage instanceof PrintAnnotationStorage ? printAnnotationStorage : this.annotationStorage; switch (annotationMode) { case AnnotationMode.DISABLE: renderingIntent += RenderingIntentFlag.ANNOTATIONS_DISABLE; @@ -11950,19 +16943,27 @@ class WorkerTransport { break; case AnnotationMode.ENABLE_STORAGE: renderingIntent += RenderingIntentFlag.ANNOTATIONS_STORAGE; - const annotationStorage = renderingIntent & RenderingIntentFlag.PRINT && printAnnotationStorage instanceof PrintAnnotationStorage ? printAnnotationStorage : this.annotationStorage; annotationStorageSerializable = annotationStorage.serializable; break; default: warn(`getRenderingIntent - invalid annotationMode: ${annotationMode}`); } + if (isEditing) { + renderingIntent += RenderingIntentFlag.IS_EDITING; + } if (isOpList) { renderingIntent += RenderingIntentFlag.OPLIST; } + const { + ids: modifiedIds, + hash: modifiedIdsHash + } = annotationStorage.modifiedIds; + const cacheKeyBuf = [renderingIntent, annotationStorageSerializable.hash, modifiedIdsHash]; return { renderingIntent, - cacheKey: `${renderingIntent}_${annotationStorageSerializable.hash}`, - annotationStorageSerializable + cacheKey: cacheKeyBuf.join("_"), + annotationStorageSerializable, + modifiedIds }; } destroy() { @@ -12657,10 +17658,13 @@ class InternalRenderTask { } } } -const version = "4.4.168"; -const build = "19fbc8998"; +const version = "4.5.136"; +const build = "3a21f03b0"; +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.flat-map.js +var esnext_iterator_flat_map = __webpack_require__(670); ;// CONCATENATED MODULE: ./src/shared/scripting_utils.js + function makeColorComp(n) { return Math.floor(Math.max(0, Math.min(1, n)) * 255).toString(16).padStart(2, "0"); } @@ -12721,6 +17725,7 @@ class ColorConverters { ;// CONCATENATED MODULE: ./src/display/xfa_layer.js + class XfaLayer { static setupStorage(html, id, element, storage, intent) { const storedData = storage.getValue(id, { @@ -12936,6 +17941,17 @@ class XfaLayer { + + + + + + + + + + + const DEFAULT_TAB_INDEX = 1000; const annotation_layer_DEFAULT_FONT_SIZE = 9; const GetElementsByNameSet = new WeakSet(); @@ -13042,6 +18058,9 @@ class AnnotationElement { }) { return !!(titleObj?.str || contentsObj?.str || richText?.str); } + get _isEditable() { + return this.data.isEditable; + } get hasPopupData() { return AnnotationElement._hasPopupData(this.data); } @@ -13502,9 +18521,6 @@ class AnnotationElement { triggers.classList.add("highlightArea"); } } - get _isEditable() { - return false; - } _editOnDoubleClick() { if (!this._isEditable) { return; @@ -14962,9 +19978,6 @@ class FreeTextAnnotationElement extends AnnotationElement { this._editOnDoubleClick(); return this.container; } - get _isEditable() { - return this.data.hasOwnCanvas; - } } class LineAnnotationElement extends AnnotationElement { #line = null; @@ -15372,6 +20385,9 @@ class AnnotationLayer { this.zIndex = 0; this._annotationEditorUIManager = annotationEditorUIManager; } + hasEditableAnnotations() { + return this.#editableAnnotations.size > 0; + } #appendElement(element, id) { const contentElement = element.firstChild || element; contentElement.id = `${AnnotationPrefix}${id}`; @@ -15438,7 +20454,7 @@ class AnnotationLayer { rendered.style.visibility = "hidden"; } this.#appendElement(rendered, data.id); - if (element.annotationEditorType > 0) { + if (element._isEditable) { this.#editableAnnotations.set(element.data.id, element); this._annotationEditorUIManager?.renderAnnotationElement(element); } @@ -15495,6 +20511,9 @@ class AnnotationLayer { + + + const EOL_PATTERN = /\r\n?|\n/g; class FreeTextEditor extends AnnotationEditor { #boundEditorDivBlur = this.editorDivBlur.bind(this); @@ -16113,6 +21132,20 @@ class FreeTextEditor extends AnnotationEditor { ;// CONCATENATED MODULE: ./src/display/editor/outliner.js + + + + + + + + + + + + + + class Outliner { #box; #verticalEdges = []; @@ -16936,6 +21969,13 @@ class ColorPicker { + + + + + + + class HighlightEditor extends AnnotationEditor { #anchorNode = null; #anchorOffset = 0; @@ -17426,6 +22466,7 @@ class HighlightEditor extends AnnotationEditor { return null; } const [pageWidth, pageHeight] = this.pageDimensions; + const [pageX, pageY] = this.pageTranslation; const boxes = this.#boxes; const quadPoints = new Float32Array(boxes.length * 8); let i = 0; @@ -17435,8 +22476,8 @@ class HighlightEditor extends AnnotationEditor { width, height } of boxes) { - const sx = x * pageWidth; - const sy = (1 - y - height) * pageHeight; + const sx = x * pageWidth + pageX; + const sy = (1 - y - height) * pageHeight + pageY; quadPoints[i] = quadPoints[i + 4] = sx; quadPoints[i + 1] = quadPoints[i + 3] = sy; quadPoints[i + 2] = quadPoints[i + 6] = sx + width * pageWidth; @@ -17576,6 +22617,8 @@ class HighlightEditor extends AnnotationEditor { + + class InkEditor extends AnnotationEditor { #baseHeight = 0; #baseWidth = 0; @@ -18390,6 +23433,9 @@ class InkEditor extends AnnotationEditor { + + + class StampEditor extends AnnotationEditor { #bitmap = null; #bitmapId = null; @@ -18398,6 +23444,7 @@ class StampEditor extends AnnotationEditor { #bitmapFile = null; #bitmapFileName = ""; #canvas = null; + #hasMLBeenQueried = false; #observer = null; #resizeTimeoutId = null; #isSvg = false; @@ -18645,6 +23692,37 @@ class StampEditor extends AnnotationEditor { } return bitmap; } + async #mlGuessAltText(bitmap, width, height) { + if (this.#hasMLBeenQueried) { + return; + } + this.#hasMLBeenQueried = true; + const isMLEnabled = await this._uiManager.isMLEnabledFor("altText"); + if (!isMLEnabled || this.hasAltText()) { + return; + } + const offscreen = new OffscreenCanvas(width, height); + const ctx = offscreen.getContext("2d", { + willReadFrequently: true + }); + ctx.drawImage(bitmap, 0, 0, bitmap.width, bitmap.height, 0, 0, width, height); + const response = await this._uiManager.mlGuess({ + service: "moz-image-to-text", + request: { + data: ctx.getImageData(0, 0, width, height).data, + width, + height, + channels: 4 + } + }); + const altText = response?.output || ""; + if (this.parent && altText && !this.hasAltText()) { + this.altTextData = { + altText, + decorative: false + }; + } + } #drawBitmap(width, height) { width = Math.ceil(width); height = Math.ceil(height); @@ -18655,28 +23733,7 @@ class StampEditor extends AnnotationEditor { canvas.width = width; canvas.height = height; const bitmap = this.#isSvg ? this.#bitmap : this.#scaleBitmap(width, height); - if (this._uiManager.hasMLManager && !this.hasAltText()) { - const offscreen = new OffscreenCanvas(width, height); - const ctx = offscreen.getContext("2d"); - ctx.drawImage(bitmap, 0, 0, bitmap.width, bitmap.height, 0, 0, width, height); - this._uiManager.mlGuess({ - service: "image-to-text", - request: { - data: ctx.getImageData(0, 0, width, height).data, - width, - height, - channels: 4 - } - }).then(response => { - const altText = response?.output || ""; - if (this.parent && altText && !this.hasAltText()) { - this.altTextData = { - altText, - decorative: false - }; - } - }); - } + this.#mlGuessAltText(bitmap, width, height); const ctx = canvas.getContext("2d"); ctx.filter = this._uiManager.hcmFilter; ctx.drawImage(bitmap, 0, 0, bitmap.width, bitmap.height, 0, 0, width, height); @@ -18815,6 +23872,14 @@ class StampEditor extends AnnotationEditor { + + + + + + + + class AnnotationEditorLayer { #accessibilityManager; #allowClick = false; @@ -18930,6 +23995,9 @@ class AnnotationEditorLayer { addCommands(params) { this.#uiManager.addCommands(params); } + toggleDrawing(enabled = false) { + this.div.classList.toggle("drawing", !enabled); + } togglePointerEvents(enabled = false) { this.div.classList.toggle("disabled", !enabled); } @@ -19057,7 +24125,10 @@ class AnnotationEditorLayer { } #textLayerPointerDown(event) { this.#uiManager.unselectAll(); - if (event.target === this.#textLayer.div) { + const { + target + } = event; + if (target === this.#textLayer.div || target.classList.contains("endOfContent") && this.#textLayer.div.contains(target)) { const { isMac } = util_FeatureTest.platform; @@ -19066,9 +24137,11 @@ class AnnotationEditorLayer { } this.#uiManager.showAllEditors("highlight", true, true); this.#textLayer.div.classList.add("free"); + this.toggleDrawing(); HighlightEditor.startHighlighting(this, this.#uiManager.direction === "ltr", event); this.#textLayer.div.addEventListener("pointerup", () => { this.#textLayer.div.classList.remove("free"); + this.toggleDrawing(true); }, { once: true, signal: this.#uiManager._signal @@ -19624,8 +24697,8 @@ class DrawLayer { -const pdfjsVersion = "4.4.168"; -const pdfjsBuild = "19fbc8998"; +const pdfjsVersion = "4.5.136"; +const pdfjsBuild = "3a21f03b0"; var __webpack_exports__AbortException = __webpack_exports__.AbortException; var __webpack_exports__AnnotationEditorLayer = __webpack_exports__.AnnotationEditorLayer; @@ -19644,7 +24717,6 @@ var __webpack_exports__ImageKind = __webpack_exports__.ImageKind; var __webpack_exports__InvalidPDFException = __webpack_exports__.InvalidPDFException; var __webpack_exports__MissingPDFException = __webpack_exports__.MissingPDFException; var __webpack_exports__OPS = __webpack_exports__.OPS; -var __webpack_exports__Outliner = __webpack_exports__.Outliner; var __webpack_exports__PDFDataRangeTransport = __webpack_exports__.PDFDataRangeTransport; var __webpack_exports__PDFDateString = __webpack_exports__.PDFDateString; var __webpack_exports__PDFWorker = __webpack_exports__.PDFWorker; @@ -19668,11 +24740,9 @@ var __webpack_exports__isDataScheme = __webpack_exports__.isDataScheme; var __webpack_exports__isPdfFile = __webpack_exports__.isPdfFile; var __webpack_exports__noContextMenu = __webpack_exports__.noContextMenu; var __webpack_exports__normalizeUnicode = __webpack_exports__.normalizeUnicode; -var __webpack_exports__renderTextLayer = __webpack_exports__.renderTextLayer; var __webpack_exports__setLayerDimensions = __webpack_exports__.setLayerDimensions; var __webpack_exports__shadow = __webpack_exports__.shadow; -var __webpack_exports__updateTextLayer = __webpack_exports__.updateTextLayer; var __webpack_exports__version = __webpack_exports__.version; -export { __webpack_exports__AbortException as AbortException, __webpack_exports__AnnotationEditorLayer as AnnotationEditorLayer, __webpack_exports__AnnotationEditorParamsType as AnnotationEditorParamsType, __webpack_exports__AnnotationEditorType as AnnotationEditorType, __webpack_exports__AnnotationEditorUIManager as AnnotationEditorUIManager, __webpack_exports__AnnotationLayer as AnnotationLayer, __webpack_exports__AnnotationMode as AnnotationMode, __webpack_exports__CMapCompressionType as CMapCompressionType, __webpack_exports__ColorPicker as ColorPicker, __webpack_exports__DOMSVGFactory as DOMSVGFactory, __webpack_exports__DrawLayer as DrawLayer, __webpack_exports__FeatureTest as FeatureTest, __webpack_exports__GlobalWorkerOptions as GlobalWorkerOptions, __webpack_exports__ImageKind as ImageKind, __webpack_exports__InvalidPDFException as InvalidPDFException, __webpack_exports__MissingPDFException as MissingPDFException, __webpack_exports__OPS as OPS, __webpack_exports__Outliner as Outliner, __webpack_exports__PDFDataRangeTransport as PDFDataRangeTransport, __webpack_exports__PDFDateString as PDFDateString, __webpack_exports__PDFWorker as PDFWorker, __webpack_exports__PasswordResponses as PasswordResponses, __webpack_exports__PermissionFlag as PermissionFlag, __webpack_exports__PixelsPerInch as PixelsPerInch, __webpack_exports__RenderingCancelledException as RenderingCancelledException, __webpack_exports__TextLayer as TextLayer, __webpack_exports__UnexpectedResponseException as UnexpectedResponseException, __webpack_exports__Util as Util, __webpack_exports__VerbosityLevel as VerbosityLevel, __webpack_exports__XfaLayer as XfaLayer, __webpack_exports__build as build, __webpack_exports__createValidAbsoluteUrl as createValidAbsoluteUrl, __webpack_exports__fetchData as fetchData, __webpack_exports__getDocument as getDocument, __webpack_exports__getFilenameFromUrl as getFilenameFromUrl, __webpack_exports__getPdfFilenameFromUrl as getPdfFilenameFromUrl, __webpack_exports__getXfaPageViewport as getXfaPageViewport, __webpack_exports__isDataScheme as isDataScheme, __webpack_exports__isPdfFile as isPdfFile, __webpack_exports__noContextMenu as noContextMenu, __webpack_exports__normalizeUnicode as normalizeUnicode, __webpack_exports__renderTextLayer as renderTextLayer, __webpack_exports__setLayerDimensions as setLayerDimensions, __webpack_exports__shadow as shadow, __webpack_exports__updateTextLayer as updateTextLayer, __webpack_exports__version as version }; +export { __webpack_exports__AbortException as AbortException, __webpack_exports__AnnotationEditorLayer as AnnotationEditorLayer, __webpack_exports__AnnotationEditorParamsType as AnnotationEditorParamsType, __webpack_exports__AnnotationEditorType as AnnotationEditorType, __webpack_exports__AnnotationEditorUIManager as AnnotationEditorUIManager, __webpack_exports__AnnotationLayer as AnnotationLayer, __webpack_exports__AnnotationMode as AnnotationMode, __webpack_exports__CMapCompressionType as CMapCompressionType, __webpack_exports__ColorPicker as ColorPicker, __webpack_exports__DOMSVGFactory as DOMSVGFactory, __webpack_exports__DrawLayer as DrawLayer, __webpack_exports__FeatureTest as FeatureTest, __webpack_exports__GlobalWorkerOptions as GlobalWorkerOptions, __webpack_exports__ImageKind as ImageKind, __webpack_exports__InvalidPDFException as InvalidPDFException, __webpack_exports__MissingPDFException as MissingPDFException, __webpack_exports__OPS as OPS, __webpack_exports__PDFDataRangeTransport as PDFDataRangeTransport, __webpack_exports__PDFDateString as PDFDateString, __webpack_exports__PDFWorker as PDFWorker, __webpack_exports__PasswordResponses as PasswordResponses, __webpack_exports__PermissionFlag as PermissionFlag, __webpack_exports__PixelsPerInch as PixelsPerInch, __webpack_exports__RenderingCancelledException as RenderingCancelledException, __webpack_exports__TextLayer as TextLayer, __webpack_exports__UnexpectedResponseException as UnexpectedResponseException, __webpack_exports__Util as Util, __webpack_exports__VerbosityLevel as VerbosityLevel, __webpack_exports__XfaLayer as XfaLayer, __webpack_exports__build as build, __webpack_exports__createValidAbsoluteUrl as createValidAbsoluteUrl, __webpack_exports__fetchData as fetchData, __webpack_exports__getDocument as getDocument, __webpack_exports__getFilenameFromUrl as getFilenameFromUrl, __webpack_exports__getPdfFilenameFromUrl as getPdfFilenameFromUrl, __webpack_exports__getXfaPageViewport as getXfaPageViewport, __webpack_exports__isDataScheme as isDataScheme, __webpack_exports__isPdfFile as isPdfFile, __webpack_exports__noContextMenu as noContextMenu, __webpack_exports__normalizeUnicode as normalizeUnicode, __webpack_exports__setLayerDimensions as setLayerDimensions, __webpack_exports__shadow as shadow, __webpack_exports__version as version }; -//# sourceMappingURL=pdf.mjs.map \ No newline at end of file +//# sourceMappingURL=pdf.mjs.map diff --git a/cps/static/js/libs/pdf.worker.mjs b/cps/static/js/libs/pdf.worker.js similarity index 94% rename from cps/static/js/libs/pdf.worker.mjs rename to cps/static/js/libs/pdf.worker.js index 94f1d20a..7bac155f 100644 --- a/cps/static/js/libs/pdf.worker.mjs +++ b/cps/static/js/libs/pdf.worker.js @@ -2,7 +2,7 @@ * @licstart The following is the entire license notice for the * JavaScript code in this page * - * Copyright 2023 Mozilla Foundation + * Copyright 2024 Mozilla Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -20,8 +20,4499 @@ * JavaScript code in this page */ -/******/ // The require scope -/******/ var __webpack_require__ = {}; +/******/ var __webpack_modules__ = ({ + +/***/ 9306: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); +var tryToString = __webpack_require__(6823); + +var $TypeError = TypeError; + +// `Assert: IsCallable(argument) is true` +module.exports = function (argument) { + if (isCallable(argument)) return argument; + throw new $TypeError(tryToString(argument) + ' is not a function'); +}; + + +/***/ }), + +/***/ 3506: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isPossiblePrototype = __webpack_require__(3925); + +var $String = String; +var $TypeError = TypeError; + +module.exports = function (argument) { + if (isPossiblePrototype(argument)) return argument; + throw new $TypeError("Can't set " + $String(argument) + ' as a prototype'); +}; + + +/***/ }), + +/***/ 7080: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var has = (__webpack_require__(4402).has); + +// Perform ? RequireInternalSlot(M, [[SetData]]) +module.exports = function (it) { + has(it); + return it; +}; + + +/***/ }), + +/***/ 679: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isPrototypeOf = __webpack_require__(1625); + +var $TypeError = TypeError; + +module.exports = function (it, Prototype) { + if (isPrototypeOf(Prototype, it)) return it; + throw new $TypeError('Incorrect invocation'); +}; + + +/***/ }), + +/***/ 8551: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isObject = __webpack_require__(34); + +var $String = String; +var $TypeError = TypeError; + +// `Assert: Type(argument) is Object` +module.exports = function (argument) { + if (isObject(argument)) return argument; + throw new $TypeError($String(argument) + ' is not an object'); +}; + + +/***/ }), + +/***/ 7811: +/***/ ((module) => { + + +// eslint-disable-next-line es/no-typed-arrays -- safe +module.exports = typeof ArrayBuffer != 'undefined' && typeof DataView != 'undefined'; + + +/***/ }), + +/***/ 7394: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThisAccessor = __webpack_require__(6706); +var classof = __webpack_require__(4576); + +var $TypeError = TypeError; + +// Includes +// - Perform ? RequireInternalSlot(O, [[ArrayBufferData]]). +// - If IsSharedArrayBuffer(O) is true, throw a TypeError exception. +module.exports = uncurryThisAccessor(ArrayBuffer.prototype, 'byteLength', 'get') || function (O) { + if (classof(O) !== 'ArrayBuffer') throw new $TypeError('ArrayBuffer expected'); + return O.byteLength; +}; + + +/***/ }), + +/***/ 3238: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var arrayBufferByteLength = __webpack_require__(7394); + +var slice = uncurryThis(ArrayBuffer.prototype.slice); + +module.exports = function (O) { + if (arrayBufferByteLength(O) !== 0) return false; + try { + slice(O, 0, 0); + return false; + } catch (error) { + return true; + } +}; + + +/***/ }), + +/***/ 5636: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var uncurryThis = __webpack_require__(9504); +var uncurryThisAccessor = __webpack_require__(6706); +var toIndex = __webpack_require__(7696); +var isDetached = __webpack_require__(3238); +var arrayBufferByteLength = __webpack_require__(7394); +var detachTransferable = __webpack_require__(4483); +var PROPER_STRUCTURED_CLONE_TRANSFER = __webpack_require__(1548); + +var structuredClone = global.structuredClone; +var ArrayBuffer = global.ArrayBuffer; +var DataView = global.DataView; +var TypeError = global.TypeError; +var min = Math.min; +var ArrayBufferPrototype = ArrayBuffer.prototype; +var DataViewPrototype = DataView.prototype; +var slice = uncurryThis(ArrayBufferPrototype.slice); +var isResizable = uncurryThisAccessor(ArrayBufferPrototype, 'resizable', 'get'); +var maxByteLength = uncurryThisAccessor(ArrayBufferPrototype, 'maxByteLength', 'get'); +var getInt8 = uncurryThis(DataViewPrototype.getInt8); +var setInt8 = uncurryThis(DataViewPrototype.setInt8); + +module.exports = (PROPER_STRUCTURED_CLONE_TRANSFER || detachTransferable) && function (arrayBuffer, newLength, preserveResizability) { + var byteLength = arrayBufferByteLength(arrayBuffer); + var newByteLength = newLength === undefined ? byteLength : toIndex(newLength); + var fixedLength = !isResizable || !isResizable(arrayBuffer); + var newBuffer; + if (isDetached(arrayBuffer)) throw new TypeError('ArrayBuffer is detached'); + if (PROPER_STRUCTURED_CLONE_TRANSFER) { + arrayBuffer = structuredClone(arrayBuffer, { transfer: [arrayBuffer] }); + if (byteLength === newByteLength && (preserveResizability || fixedLength)) return arrayBuffer; + } + if (byteLength >= newByteLength && (!preserveResizability || fixedLength)) { + newBuffer = slice(arrayBuffer, 0, newByteLength); + } else { + var options = preserveResizability && !fixedLength && maxByteLength ? { maxByteLength: maxByteLength(arrayBuffer) } : undefined; + newBuffer = new ArrayBuffer(newByteLength, options); + var a = new DataView(arrayBuffer); + var b = new DataView(newBuffer); + var copyLength = min(newByteLength, byteLength); + for (var i = 0; i < copyLength; i++) setInt8(b, i, getInt8(a, i)); + } + if (!PROPER_STRUCTURED_CLONE_TRANSFER) detachTransferable(arrayBuffer); + return newBuffer; +}; + + +/***/ }), + +/***/ 4644: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_ARRAY_BUFFER = __webpack_require__(7811); +var DESCRIPTORS = __webpack_require__(3724); +var global = __webpack_require__(4475); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var hasOwn = __webpack_require__(9297); +var classof = __webpack_require__(6955); +var tryToString = __webpack_require__(6823); +var createNonEnumerableProperty = __webpack_require__(6699); +var defineBuiltIn = __webpack_require__(6840); +var defineBuiltInAccessor = __webpack_require__(2106); +var isPrototypeOf = __webpack_require__(1625); +var getPrototypeOf = __webpack_require__(2787); +var setPrototypeOf = __webpack_require__(2967); +var wellKnownSymbol = __webpack_require__(8227); +var uid = __webpack_require__(3392); +var InternalStateModule = __webpack_require__(1181); + +var enforceInternalState = InternalStateModule.enforce; +var getInternalState = InternalStateModule.get; +var Int8Array = global.Int8Array; +var Int8ArrayPrototype = Int8Array && Int8Array.prototype; +var Uint8ClampedArray = global.Uint8ClampedArray; +var Uint8ClampedArrayPrototype = Uint8ClampedArray && Uint8ClampedArray.prototype; +var TypedArray = Int8Array && getPrototypeOf(Int8Array); +var TypedArrayPrototype = Int8ArrayPrototype && getPrototypeOf(Int8ArrayPrototype); +var ObjectPrototype = Object.prototype; +var TypeError = global.TypeError; + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var TYPED_ARRAY_TAG = uid('TYPED_ARRAY_TAG'); +var TYPED_ARRAY_CONSTRUCTOR = 'TypedArrayConstructor'; +// Fixing native typed arrays in Opera Presto crashes the browser, see #595 +var NATIVE_ARRAY_BUFFER_VIEWS = NATIVE_ARRAY_BUFFER && !!setPrototypeOf && classof(global.opera) !== 'Opera'; +var TYPED_ARRAY_TAG_REQUIRED = false; +var NAME, Constructor, Prototype; + +var TypedArrayConstructorsList = { + Int8Array: 1, + Uint8Array: 1, + Uint8ClampedArray: 1, + Int16Array: 2, + Uint16Array: 2, + Int32Array: 4, + Uint32Array: 4, + Float32Array: 4, + Float64Array: 8 +}; + +var BigIntArrayConstructorsList = { + BigInt64Array: 8, + BigUint64Array: 8 +}; + +var isView = function isView(it) { + if (!isObject(it)) return false; + var klass = classof(it); + return klass === 'DataView' + || hasOwn(TypedArrayConstructorsList, klass) + || hasOwn(BigIntArrayConstructorsList, klass); +}; + +var getTypedArrayConstructor = function (it) { + var proto = getPrototypeOf(it); + if (!isObject(proto)) return; + var state = getInternalState(proto); + return (state && hasOwn(state, TYPED_ARRAY_CONSTRUCTOR)) ? state[TYPED_ARRAY_CONSTRUCTOR] : getTypedArrayConstructor(proto); +}; + +var isTypedArray = function (it) { + if (!isObject(it)) return false; + var klass = classof(it); + return hasOwn(TypedArrayConstructorsList, klass) + || hasOwn(BigIntArrayConstructorsList, klass); +}; + +var aTypedArray = function (it) { + if (isTypedArray(it)) return it; + throw new TypeError('Target is not a typed array'); +}; + +var aTypedArrayConstructor = function (C) { + if (isCallable(C) && (!setPrototypeOf || isPrototypeOf(TypedArray, C))) return C; + throw new TypeError(tryToString(C) + ' is not a typed array constructor'); +}; + +var exportTypedArrayMethod = function (KEY, property, forced, options) { + if (!DESCRIPTORS) return; + if (forced) for (var ARRAY in TypedArrayConstructorsList) { + var TypedArrayConstructor = global[ARRAY]; + if (TypedArrayConstructor && hasOwn(TypedArrayConstructor.prototype, KEY)) try { + delete TypedArrayConstructor.prototype[KEY]; + } catch (error) { + // old WebKit bug - some methods are non-configurable + try { + TypedArrayConstructor.prototype[KEY] = property; + } catch (error2) { /* empty */ } + } + } + if (!TypedArrayPrototype[KEY] || forced) { + defineBuiltIn(TypedArrayPrototype, KEY, forced ? property + : NATIVE_ARRAY_BUFFER_VIEWS && Int8ArrayPrototype[KEY] || property, options); + } +}; + +var exportTypedArrayStaticMethod = function (KEY, property, forced) { + var ARRAY, TypedArrayConstructor; + if (!DESCRIPTORS) return; + if (setPrototypeOf) { + if (forced) for (ARRAY in TypedArrayConstructorsList) { + TypedArrayConstructor = global[ARRAY]; + if (TypedArrayConstructor && hasOwn(TypedArrayConstructor, KEY)) try { + delete TypedArrayConstructor[KEY]; + } catch (error) { /* empty */ } + } + if (!TypedArray[KEY] || forced) { + // V8 ~ Chrome 49-50 `%TypedArray%` methods are non-writable non-configurable + try { + return defineBuiltIn(TypedArray, KEY, forced ? property : NATIVE_ARRAY_BUFFER_VIEWS && TypedArray[KEY] || property); + } catch (error) { /* empty */ } + } else return; + } + for (ARRAY in TypedArrayConstructorsList) { + TypedArrayConstructor = global[ARRAY]; + if (TypedArrayConstructor && (!TypedArrayConstructor[KEY] || forced)) { + defineBuiltIn(TypedArrayConstructor, KEY, property); + } + } +}; + +for (NAME in TypedArrayConstructorsList) { + Constructor = global[NAME]; + Prototype = Constructor && Constructor.prototype; + if (Prototype) enforceInternalState(Prototype)[TYPED_ARRAY_CONSTRUCTOR] = Constructor; + else NATIVE_ARRAY_BUFFER_VIEWS = false; +} + +for (NAME in BigIntArrayConstructorsList) { + Constructor = global[NAME]; + Prototype = Constructor && Constructor.prototype; + if (Prototype) enforceInternalState(Prototype)[TYPED_ARRAY_CONSTRUCTOR] = Constructor; +} + +// WebKit bug - typed arrays constructors prototype is Object.prototype +if (!NATIVE_ARRAY_BUFFER_VIEWS || !isCallable(TypedArray) || TypedArray === Function.prototype) { + // eslint-disable-next-line no-shadow -- safe + TypedArray = function TypedArray() { + throw new TypeError('Incorrect invocation'); + }; + if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) { + if (global[NAME]) setPrototypeOf(global[NAME], TypedArray); + } +} + +if (!NATIVE_ARRAY_BUFFER_VIEWS || !TypedArrayPrototype || TypedArrayPrototype === ObjectPrototype) { + TypedArrayPrototype = TypedArray.prototype; + if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) { + if (global[NAME]) setPrototypeOf(global[NAME].prototype, TypedArrayPrototype); + } +} + +// WebKit bug - one more object in Uint8ClampedArray prototype chain +if (NATIVE_ARRAY_BUFFER_VIEWS && getPrototypeOf(Uint8ClampedArrayPrototype) !== TypedArrayPrototype) { + setPrototypeOf(Uint8ClampedArrayPrototype, TypedArrayPrototype); +} + +if (DESCRIPTORS && !hasOwn(TypedArrayPrototype, TO_STRING_TAG)) { + TYPED_ARRAY_TAG_REQUIRED = true; + defineBuiltInAccessor(TypedArrayPrototype, TO_STRING_TAG, { + configurable: true, + get: function () { + return isObject(this) ? this[TYPED_ARRAY_TAG] : undefined; + } + }); + for (NAME in TypedArrayConstructorsList) if (global[NAME]) { + createNonEnumerableProperty(global[NAME], TYPED_ARRAY_TAG, NAME); + } +} + +module.exports = { + NATIVE_ARRAY_BUFFER_VIEWS: NATIVE_ARRAY_BUFFER_VIEWS, + TYPED_ARRAY_TAG: TYPED_ARRAY_TAG_REQUIRED && TYPED_ARRAY_TAG, + aTypedArray: aTypedArray, + aTypedArrayConstructor: aTypedArrayConstructor, + exportTypedArrayMethod: exportTypedArrayMethod, + exportTypedArrayStaticMethod: exportTypedArrayStaticMethod, + getTypedArrayConstructor: getTypedArrayConstructor, + isView: isView, + isTypedArray: isTypedArray, + TypedArray: TypedArray, + TypedArrayPrototype: TypedArrayPrototype +}; + + +/***/ }), + +/***/ 5370: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var lengthOfArrayLike = __webpack_require__(6198); + +module.exports = function (Constructor, list, $length) { + var index = 0; + var length = arguments.length > 2 ? $length : lengthOfArrayLike(list); + var result = new Constructor(length); + while (length > index) result[index] = list[index++]; + return result; +}; + + +/***/ }), + +/***/ 9617: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIndexedObject = __webpack_require__(5397); +var toAbsoluteIndex = __webpack_require__(5610); +var lengthOfArrayLike = __webpack_require__(6198); + +// `Array.prototype.{ indexOf, includes }` methods implementation +var createMethod = function (IS_INCLUDES) { + return function ($this, el, fromIndex) { + var O = toIndexedObject($this); + var length = lengthOfArrayLike(O); + if (length === 0) return !IS_INCLUDES && -1; + var index = toAbsoluteIndex(fromIndex, length); + var value; + // Array#includes uses SameValueZero equality algorithm + // eslint-disable-next-line no-self-compare -- NaN check + if (IS_INCLUDES && el !== el) while (length > index) { + value = O[index++]; + // eslint-disable-next-line no-self-compare -- NaN check + if (value !== value) return true; + // Array#indexOf ignores holes, Array#includes - not + } else for (;length > index; index++) { + if ((IS_INCLUDES || index in O) && O[index] === el) return IS_INCLUDES || index || 0; + } return !IS_INCLUDES && -1; + }; +}; + +module.exports = { + // `Array.prototype.includes` method + // https://tc39.es/ecma262/#sec-array.prototype.includes + includes: createMethod(true), + // `Array.prototype.indexOf` method + // https://tc39.es/ecma262/#sec-array.prototype.indexof + indexOf: createMethod(false) +}; + + +/***/ }), + +/***/ 4527: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var isArray = __webpack_require__(4376); + +var $TypeError = TypeError; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; + +// Safari < 13 does not throw an error in this case +var SILENT_ON_NON_WRITABLE_LENGTH_SET = DESCRIPTORS && !function () { + // makes no sense without proper strict mode support + if (this !== undefined) return true; + try { + // eslint-disable-next-line es/no-object-defineproperty -- safe + Object.defineProperty([], 'length', { writable: false }).length = 1; + } catch (error) { + return error instanceof TypeError; + } +}(); + +module.exports = SILENT_ON_NON_WRITABLE_LENGTH_SET ? function (O, length) { + if (isArray(O) && !getOwnPropertyDescriptor(O, 'length').writable) { + throw new $TypeError('Cannot set read only .length'); + } return O.length = length; +} : function (O, length) { + return O.length = length; +}; + + +/***/ }), + +/***/ 7628: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var lengthOfArrayLike = __webpack_require__(6198); + +// https://tc39.es/proposal-change-array-by-copy/#sec-array.prototype.toReversed +// https://tc39.es/proposal-change-array-by-copy/#sec-%typedarray%.prototype.toReversed +module.exports = function (O, C) { + var len = lengthOfArrayLike(O); + var A = new C(len); + var k = 0; + for (; k < len; k++) A[k] = O[len - k - 1]; + return A; +}; + + +/***/ }), + +/***/ 9928: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var lengthOfArrayLike = __webpack_require__(6198); +var toIntegerOrInfinity = __webpack_require__(1291); + +var $RangeError = RangeError; + +// https://tc39.es/proposal-change-array-by-copy/#sec-array.prototype.with +// https://tc39.es/proposal-change-array-by-copy/#sec-%typedarray%.prototype.with +module.exports = function (O, C, index, value) { + var len = lengthOfArrayLike(O); + var relativeIndex = toIntegerOrInfinity(index); + var actualIndex = relativeIndex < 0 ? len + relativeIndex : relativeIndex; + if (actualIndex >= len || actualIndex < 0) throw new $RangeError('Incorrect index'); + var A = new C(len); + var k = 0; + for (; k < len; k++) A[k] = k === actualIndex ? value : O[k]; + return A; +}; + + +/***/ }), + +/***/ 6319: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var anObject = __webpack_require__(8551); +var iteratorClose = __webpack_require__(9539); + +// call something on iterator step with safe closing on error +module.exports = function (iterator, fn, value, ENTRIES) { + try { + return ENTRIES ? fn(anObject(value)[0], value[1]) : fn(value); + } catch (error) { + iteratorClose(iterator, 'throw', error); + } +}; + + +/***/ }), + +/***/ 4576: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +var toString = uncurryThis({}.toString); +var stringSlice = uncurryThis(''.slice); + +module.exports = function (it) { + return stringSlice(toString(it), 8, -1); +}; + + +/***/ }), + +/***/ 6955: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var TO_STRING_TAG_SUPPORT = __webpack_require__(2140); +var isCallable = __webpack_require__(4901); +var classofRaw = __webpack_require__(4576); +var wellKnownSymbol = __webpack_require__(8227); + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var $Object = Object; + +// ES3 wrong here +var CORRECT_ARGUMENTS = classofRaw(function () { return arguments; }()) === 'Arguments'; + +// fallback for IE11 Script Access Denied error +var tryGet = function (it, key) { + try { + return it[key]; + } catch (error) { /* empty */ } +}; + +// getting tag from ES6+ `Object.prototype.toString` +module.exports = TO_STRING_TAG_SUPPORT ? classofRaw : function (it) { + var O, tag, result; + return it === undefined ? 'Undefined' : it === null ? 'Null' + // @@toStringTag case + : typeof (tag = tryGet(O = $Object(it), TO_STRING_TAG)) == 'string' ? tag + // builtinTag case + : CORRECT_ARGUMENTS ? classofRaw(O) + // ES3 arguments fallback + : (result = classofRaw(O)) === 'Object' && isCallable(O.callee) ? 'Arguments' : result; +}; + + +/***/ }), + +/***/ 7740: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var hasOwn = __webpack_require__(9297); +var ownKeys = __webpack_require__(5031); +var getOwnPropertyDescriptorModule = __webpack_require__(7347); +var definePropertyModule = __webpack_require__(4913); + +module.exports = function (target, source, exceptions) { + var keys = ownKeys(source); + var defineProperty = definePropertyModule.f; + var getOwnPropertyDescriptor = getOwnPropertyDescriptorModule.f; + for (var i = 0; i < keys.length; i++) { + var key = keys[i]; + if (!hasOwn(target, key) && !(exceptions && hasOwn(exceptions, key))) { + defineProperty(target, key, getOwnPropertyDescriptor(source, key)); + } + } +}; + + +/***/ }), + +/***/ 2211: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); + +module.exports = !fails(function () { + function F() { /* empty */ } + F.prototype.constructor = null; + // eslint-disable-next-line es/no-object-getprototypeof -- required for testing + return Object.getPrototypeOf(new F()) !== F.prototype; +}); + + +/***/ }), + +/***/ 2529: +/***/ ((module) => { + + +// `CreateIterResultObject` abstract operation +// https://tc39.es/ecma262/#sec-createiterresultobject +module.exports = function (value, done) { + return { value: value, done: done }; +}; + + +/***/ }), + +/***/ 6699: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var definePropertyModule = __webpack_require__(4913); +var createPropertyDescriptor = __webpack_require__(6980); + +module.exports = DESCRIPTORS ? function (object, key, value) { + return definePropertyModule.f(object, key, createPropertyDescriptor(1, value)); +} : function (object, key, value) { + object[key] = value; + return object; +}; + + +/***/ }), + +/***/ 6980: +/***/ ((module) => { + + +module.exports = function (bitmap, value) { + return { + enumerable: !(bitmap & 1), + configurable: !(bitmap & 2), + writable: !(bitmap & 4), + value: value + }; +}; + + +/***/ }), + +/***/ 4659: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var definePropertyModule = __webpack_require__(4913); +var createPropertyDescriptor = __webpack_require__(6980); + +module.exports = function (object, key, value) { + if (DESCRIPTORS) definePropertyModule.f(object, key, createPropertyDescriptor(0, value)); + else object[key] = value; +}; + + +/***/ }), + +/***/ 2106: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var makeBuiltIn = __webpack_require__(283); +var defineProperty = __webpack_require__(4913); + +module.exports = function (target, name, descriptor) { + if (descriptor.get) makeBuiltIn(descriptor.get, name, { getter: true }); + if (descriptor.set) makeBuiltIn(descriptor.set, name, { setter: true }); + return defineProperty.f(target, name, descriptor); +}; + + +/***/ }), + +/***/ 6840: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); +var definePropertyModule = __webpack_require__(4913); +var makeBuiltIn = __webpack_require__(283); +var defineGlobalProperty = __webpack_require__(9433); + +module.exports = function (O, key, value, options) { + if (!options) options = {}; + var simple = options.enumerable; + var name = options.name !== undefined ? options.name : key; + if (isCallable(value)) makeBuiltIn(value, name, options); + if (options.global) { + if (simple) O[key] = value; + else defineGlobalProperty(key, value); + } else { + try { + if (!options.unsafe) delete O[key]; + else if (O[key]) simple = true; + } catch (error) { /* empty */ } + if (simple) O[key] = value; + else definePropertyModule.f(O, key, { + value: value, + enumerable: false, + configurable: !options.nonConfigurable, + writable: !options.nonWritable + }); + } return O; +}; + + +/***/ }), + +/***/ 6279: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var defineBuiltIn = __webpack_require__(6840); + +module.exports = function (target, src, options) { + for (var key in src) defineBuiltIn(target, key, src[key], options); + return target; +}; + + +/***/ }), + +/***/ 9433: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); + +// eslint-disable-next-line es/no-object-defineproperty -- safe +var defineProperty = Object.defineProperty; + +module.exports = function (key, value) { + try { + defineProperty(global, key, { value: value, configurable: true, writable: true }); + } catch (error) { + global[key] = value; + } return value; +}; + + +/***/ }), + +/***/ 3724: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); + +// Detect IE8's incomplete defineProperty implementation +module.exports = !fails(function () { + // eslint-disable-next-line es/no-object-defineproperty -- required for testing + return Object.defineProperty({}, 1, { get: function () { return 7; } })[1] !== 7; +}); + + +/***/ }), + +/***/ 4483: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var tryNodeRequire = __webpack_require__(9714); +var PROPER_STRUCTURED_CLONE_TRANSFER = __webpack_require__(1548); + +var structuredClone = global.structuredClone; +var $ArrayBuffer = global.ArrayBuffer; +var $MessageChannel = global.MessageChannel; +var detach = false; +var WorkerThreads, channel, buffer, $detach; + +if (PROPER_STRUCTURED_CLONE_TRANSFER) { + detach = function (transferable) { + structuredClone(transferable, { transfer: [transferable] }); + }; +} else if ($ArrayBuffer) try { + if (!$MessageChannel) { + WorkerThreads = tryNodeRequire('worker_threads'); + if (WorkerThreads) $MessageChannel = WorkerThreads.MessageChannel; + } + + if ($MessageChannel) { + channel = new $MessageChannel(); + buffer = new $ArrayBuffer(2); + + $detach = function (transferable) { + channel.port1.postMessage(null, [transferable]); + }; + + if (buffer.byteLength === 2) { + $detach(buffer); + if (buffer.byteLength === 0) detach = $detach; + } + } +} catch (error) { /* empty */ } + +module.exports = detach; + + +/***/ }), + +/***/ 4055: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var isObject = __webpack_require__(34); + +var document = global.document; +// typeof document.createElement is 'object' in old IE +var EXISTS = isObject(document) && isObject(document.createElement); + +module.exports = function (it) { + return EXISTS ? document.createElement(it) : {}; +}; + + +/***/ }), + +/***/ 6837: +/***/ ((module) => { + + +var $TypeError = TypeError; +var MAX_SAFE_INTEGER = 0x1FFFFFFFFFFFFF; // 2 ** 53 - 1 == 9007199254740991 + +module.exports = function (it) { + if (it > MAX_SAFE_INTEGER) throw $TypeError('Maximum allowed index exceeded'); + return it; +}; + + +/***/ }), + +/***/ 5002: +/***/ ((module) => { + + +module.exports = { + IndexSizeError: { s: 'INDEX_SIZE_ERR', c: 1, m: 1 }, + DOMStringSizeError: { s: 'DOMSTRING_SIZE_ERR', c: 2, m: 0 }, + HierarchyRequestError: { s: 'HIERARCHY_REQUEST_ERR', c: 3, m: 1 }, + WrongDocumentError: { s: 'WRONG_DOCUMENT_ERR', c: 4, m: 1 }, + InvalidCharacterError: { s: 'INVALID_CHARACTER_ERR', c: 5, m: 1 }, + NoDataAllowedError: { s: 'NO_DATA_ALLOWED_ERR', c: 6, m: 0 }, + NoModificationAllowedError: { s: 'NO_MODIFICATION_ALLOWED_ERR', c: 7, m: 1 }, + NotFoundError: { s: 'NOT_FOUND_ERR', c: 8, m: 1 }, + NotSupportedError: { s: 'NOT_SUPPORTED_ERR', c: 9, m: 1 }, + InUseAttributeError: { s: 'INUSE_ATTRIBUTE_ERR', c: 10, m: 1 }, + InvalidStateError: { s: 'INVALID_STATE_ERR', c: 11, m: 1 }, + SyntaxError: { s: 'SYNTAX_ERR', c: 12, m: 1 }, + InvalidModificationError: { s: 'INVALID_MODIFICATION_ERR', c: 13, m: 1 }, + NamespaceError: { s: 'NAMESPACE_ERR', c: 14, m: 1 }, + InvalidAccessError: { s: 'INVALID_ACCESS_ERR', c: 15, m: 1 }, + ValidationError: { s: 'VALIDATION_ERR', c: 16, m: 0 }, + TypeMismatchError: { s: 'TYPE_MISMATCH_ERR', c: 17, m: 1 }, + SecurityError: { s: 'SECURITY_ERR', c: 18, m: 1 }, + NetworkError: { s: 'NETWORK_ERR', c: 19, m: 1 }, + AbortError: { s: 'ABORT_ERR', c: 20, m: 1 }, + URLMismatchError: { s: 'URL_MISMATCH_ERR', c: 21, m: 1 }, + QuotaExceededError: { s: 'QUOTA_EXCEEDED_ERR', c: 22, m: 1 }, + TimeoutError: { s: 'TIMEOUT_ERR', c: 23, m: 1 }, + InvalidNodeTypeError: { s: 'INVALID_NODE_TYPE_ERR', c: 24, m: 1 }, + DataCloneError: { s: 'DATA_CLONE_ERR', c: 25, m: 1 } +}; + + +/***/ }), + +/***/ 7290: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var IS_DENO = __webpack_require__(516); +var IS_NODE = __webpack_require__(9088); + +module.exports = !IS_DENO && !IS_NODE + && typeof window == 'object' + && typeof document == 'object'; + + +/***/ }), + +/***/ 516: +/***/ ((module) => { + + +/* global Deno -- Deno case */ +module.exports = typeof Deno == 'object' && Deno && typeof Deno.version == 'object'; + + +/***/ }), + +/***/ 9088: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var classof = __webpack_require__(4576); + +module.exports = classof(global.process) === 'process'; + + +/***/ }), + +/***/ 9392: +/***/ ((module) => { + + +module.exports = typeof navigator != 'undefined' && String(navigator.userAgent) || ''; + + +/***/ }), + +/***/ 7388: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var userAgent = __webpack_require__(9392); + +var process = global.process; +var Deno = global.Deno; +var versions = process && process.versions || Deno && Deno.version; +var v8 = versions && versions.v8; +var match, version; + +if (v8) { + match = v8.split('.'); + // in old Chrome, versions of V8 isn't V8 = Chrome / 10 + // but their correct versions are not interesting for us + version = match[0] > 0 && match[0] < 4 ? 1 : +(match[0] + match[1]); +} + +// BrowserFS NodeJS `process` polyfill incorrectly set `.v8` to `0.0` +// so check `userAgent` even if `.v8` exists, but 0 +if (!version && userAgent) { + match = userAgent.match(/Edge\/(\d+)/); + if (!match || match[1] >= 74) { + match = userAgent.match(/Chrome\/(\d+)/); + if (match) version = +match[1]; + } +} + +module.exports = version; + + +/***/ }), + +/***/ 8727: +/***/ ((module) => { + + +// IE8- don't enum bug keys +module.exports = [ + 'constructor', + 'hasOwnProperty', + 'isPrototypeOf', + 'propertyIsEnumerable', + 'toLocaleString', + 'toString', + 'valueOf' +]; + + +/***/ }), + +/***/ 6193: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +var $Error = Error; +var replace = uncurryThis(''.replace); + +var TEST = (function (arg) { return String(new $Error(arg).stack); })('zxcasd'); +// eslint-disable-next-line redos/no-vulnerable -- safe +var V8_OR_CHAKRA_STACK_ENTRY = /\n\s*at [^:]*:[^\n]*/; +var IS_V8_OR_CHAKRA_STACK = V8_OR_CHAKRA_STACK_ENTRY.test(TEST); + +module.exports = function (stack, dropEntries) { + if (IS_V8_OR_CHAKRA_STACK && typeof stack == 'string' && !$Error.prepareStackTrace) { + while (dropEntries--) stack = replace(stack, V8_OR_CHAKRA_STACK_ENTRY, ''); + } return stack; +}; + + +/***/ }), + +/***/ 6518: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var getOwnPropertyDescriptor = (__webpack_require__(7347).f); +var createNonEnumerableProperty = __webpack_require__(6699); +var defineBuiltIn = __webpack_require__(6840); +var defineGlobalProperty = __webpack_require__(9433); +var copyConstructorProperties = __webpack_require__(7740); +var isForced = __webpack_require__(2796); + +/* + options.target - name of the target object + options.global - target is the global object + options.stat - export as static methods of target + options.proto - export as prototype methods of target + options.real - real prototype method for the `pure` version + options.forced - export even if the native feature is available + options.bind - bind methods to the target, required for the `pure` version + options.wrap - wrap constructors to preventing global pollution, required for the `pure` version + options.unsafe - use the simple assignment of property instead of delete + defineProperty + options.sham - add a flag to not completely full polyfills + options.enumerable - export as enumerable property + options.dontCallGetSet - prevent calling a getter on target + options.name - the .name of the function if it does not match the key +*/ +module.exports = function (options, source) { + var TARGET = options.target; + var GLOBAL = options.global; + var STATIC = options.stat; + var FORCED, target, key, targetProperty, sourceProperty, descriptor; + if (GLOBAL) { + target = global; + } else if (STATIC) { + target = global[TARGET] || defineGlobalProperty(TARGET, {}); + } else { + target = global[TARGET] && global[TARGET].prototype; + } + if (target) for (key in source) { + sourceProperty = source[key]; + if (options.dontCallGetSet) { + descriptor = getOwnPropertyDescriptor(target, key); + targetProperty = descriptor && descriptor.value; + } else targetProperty = target[key]; + FORCED = isForced(GLOBAL ? key : TARGET + (STATIC ? '.' : '#') + key, options.forced); + // contained in target + if (!FORCED && targetProperty !== undefined) { + if (typeof sourceProperty == typeof targetProperty) continue; + copyConstructorProperties(sourceProperty, targetProperty); + } + // add a flag to not completely full polyfills + if (options.sham || (targetProperty && targetProperty.sham)) { + createNonEnumerableProperty(sourceProperty, 'sham', true); + } + defineBuiltIn(target, key, sourceProperty, options); + } +}; + + +/***/ }), + +/***/ 9039: +/***/ ((module) => { + + +module.exports = function (exec) { + try { + return !!exec(); + } catch (error) { + return true; + } +}; + + +/***/ }), + +/***/ 6080: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(7476); +var aCallable = __webpack_require__(9306); +var NATIVE_BIND = __webpack_require__(616); + +var bind = uncurryThis(uncurryThis.bind); + +// optional / simple context binding +module.exports = function (fn, that) { + aCallable(fn); + return that === undefined ? fn : NATIVE_BIND ? bind(fn, that) : function (/* ...args */) { + return fn.apply(that, arguments); + }; +}; + + +/***/ }), + +/***/ 616: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); + +module.exports = !fails(function () { + // eslint-disable-next-line es/no-function-prototype-bind -- safe + var test = (function () { /* empty */ }).bind(); + // eslint-disable-next-line no-prototype-builtins -- safe + return typeof test != 'function' || test.hasOwnProperty('prototype'); +}); + + +/***/ }), + +/***/ 9565: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_BIND = __webpack_require__(616); + +var call = Function.prototype.call; + +module.exports = NATIVE_BIND ? call.bind(call) : function () { + return call.apply(call, arguments); +}; + + +/***/ }), + +/***/ 350: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var hasOwn = __webpack_require__(9297); + +var FunctionPrototype = Function.prototype; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getDescriptor = DESCRIPTORS && Object.getOwnPropertyDescriptor; + +var EXISTS = hasOwn(FunctionPrototype, 'name'); +// additional protection from minified / mangled / dropped function names +var PROPER = EXISTS && (function something() { /* empty */ }).name === 'something'; +var CONFIGURABLE = EXISTS && (!DESCRIPTORS || (DESCRIPTORS && getDescriptor(FunctionPrototype, 'name').configurable)); + +module.exports = { + EXISTS: EXISTS, + PROPER: PROPER, + CONFIGURABLE: CONFIGURABLE +}; + + +/***/ }), + +/***/ 6706: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var aCallable = __webpack_require__(9306); + +module.exports = function (object, key, method) { + try { + // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe + return uncurryThis(aCallable(Object.getOwnPropertyDescriptor(object, key)[method])); + } catch (error) { /* empty */ } +}; + + +/***/ }), + +/***/ 7476: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classofRaw = __webpack_require__(4576); +var uncurryThis = __webpack_require__(9504); + +module.exports = function (fn) { + // Nashorn bug: + // https://github.com/zloirock/core-js/issues/1128 + // https://github.com/zloirock/core-js/issues/1130 + if (classofRaw(fn) === 'Function') return uncurryThis(fn); +}; + + +/***/ }), + +/***/ 9504: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_BIND = __webpack_require__(616); + +var FunctionPrototype = Function.prototype; +var call = FunctionPrototype.call; +var uncurryThisWithBind = NATIVE_BIND && FunctionPrototype.bind.bind(call, call); + +module.exports = NATIVE_BIND ? uncurryThisWithBind : function (fn) { + return function () { + return call.apply(fn, arguments); + }; +}; + + +/***/ }), + +/***/ 7751: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var isCallable = __webpack_require__(4901); + +var aFunction = function (argument) { + return isCallable(argument) ? argument : undefined; +}; + +module.exports = function (namespace, method) { + return arguments.length < 2 ? aFunction(global[namespace]) : global[namespace] && global[namespace][method]; +}; + + +/***/ }), + +/***/ 1767: +/***/ ((module) => { + + +// `GetIteratorDirect(obj)` abstract operation +// https://tc39.es/proposal-iterator-helpers/#sec-getiteratordirect +module.exports = function (obj) { + return { + iterator: obj, + next: obj.next, + done: false + }; +}; + + +/***/ }), + +/***/ 851: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(6955); +var getMethod = __webpack_require__(5966); +var isNullOrUndefined = __webpack_require__(4117); +var Iterators = __webpack_require__(6269); +var wellKnownSymbol = __webpack_require__(8227); + +var ITERATOR = wellKnownSymbol('iterator'); + +module.exports = function (it) { + if (!isNullOrUndefined(it)) return getMethod(it, ITERATOR) + || getMethod(it, '@@iterator') + || Iterators[classof(it)]; +}; + + +/***/ }), + +/***/ 81: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var tryToString = __webpack_require__(6823); +var getIteratorMethod = __webpack_require__(851); + +var $TypeError = TypeError; + +module.exports = function (argument, usingIterator) { + var iteratorMethod = arguments.length < 2 ? getIteratorMethod(argument) : usingIterator; + if (aCallable(iteratorMethod)) return anObject(call(iteratorMethod, argument)); + throw new $TypeError(tryToString(argument) + ' is not iterable'); +}; + + +/***/ }), + +/***/ 5966: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aCallable = __webpack_require__(9306); +var isNullOrUndefined = __webpack_require__(4117); + +// `GetMethod` abstract operation +// https://tc39.es/ecma262/#sec-getmethod +module.exports = function (V, P) { + var func = V[P]; + return isNullOrUndefined(func) ? undefined : aCallable(func); +}; + + +/***/ }), + +/***/ 3789: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var call = __webpack_require__(9565); +var toIntegerOrInfinity = __webpack_require__(1291); +var getIteratorDirect = __webpack_require__(1767); + +var INVALID_SIZE = 'Invalid size'; +var $RangeError = RangeError; +var $TypeError = TypeError; +var max = Math.max; + +var SetRecord = function (set, intSize) { + this.set = set; + this.size = max(intSize, 0); + this.has = aCallable(set.has); + this.keys = aCallable(set.keys); +}; + +SetRecord.prototype = { + getIterator: function () { + return getIteratorDirect(anObject(call(this.keys, this.set))); + }, + includes: function (it) { + return call(this.has, this.set, it); + } +}; + +// `GetSetRecord` abstract operation +// https://tc39.es/proposal-set-methods/#sec-getsetrecord +module.exports = function (obj) { + anObject(obj); + var numSize = +obj.size; + // NOTE: If size is undefined, then numSize will be NaN + // eslint-disable-next-line no-self-compare -- NaN check + if (numSize !== numSize) throw new $TypeError(INVALID_SIZE); + var intSize = toIntegerOrInfinity(numSize); + if (intSize < 0) throw new $RangeError(INVALID_SIZE); + return new SetRecord(obj, intSize); +}; + + +/***/ }), + +/***/ 4475: +/***/ (function(module) { + + +var check = function (it) { + return it && it.Math === Math && it; +}; + +// https://github.com/zloirock/core-js/issues/86#issuecomment-115759028 +module.exports = + // eslint-disable-next-line es/no-global-this -- safe + check(typeof globalThis == 'object' && globalThis) || + check(typeof window == 'object' && window) || + // eslint-disable-next-line no-restricted-globals -- safe + check(typeof self == 'object' && self) || + check(typeof global == 'object' && global) || + check(typeof this == 'object' && this) || + // eslint-disable-next-line no-new-func -- fallback + (function () { return this; })() || Function('return this')(); + + +/***/ }), + +/***/ 9297: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var toObject = __webpack_require__(8981); + +var hasOwnProperty = uncurryThis({}.hasOwnProperty); + +// `HasOwnProperty` abstract operation +// https://tc39.es/ecma262/#sec-hasownproperty +// eslint-disable-next-line es/no-object-hasown -- safe +module.exports = Object.hasOwn || function hasOwn(it, key) { + return hasOwnProperty(toObject(it), key); +}; + + +/***/ }), + +/***/ 421: +/***/ ((module) => { + + +module.exports = {}; + + +/***/ }), + +/***/ 397: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); + +module.exports = getBuiltIn('document', 'documentElement'); + + +/***/ }), + +/***/ 5917: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var fails = __webpack_require__(9039); +var createElement = __webpack_require__(4055); + +// Thanks to IE8 for its funny defineProperty +module.exports = !DESCRIPTORS && !fails(function () { + // eslint-disable-next-line es/no-object-defineproperty -- required for testing + return Object.defineProperty(createElement('div'), 'a', { + get: function () { return 7; } + }).a !== 7; +}); + + +/***/ }), + +/***/ 7055: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var fails = __webpack_require__(9039); +var classof = __webpack_require__(4576); + +var $Object = Object; +var split = uncurryThis(''.split); + +// fallback for non-array-like ES3 and non-enumerable old V8 strings +module.exports = fails(function () { + // throws an error in rhino, see https://github.com/mozilla/rhino/issues/346 + // eslint-disable-next-line no-prototype-builtins -- safe + return !$Object('z').propertyIsEnumerable(0); +}) ? function (it) { + return classof(it) === 'String' ? split(it, '') : $Object(it); +} : $Object; + + +/***/ }), + +/***/ 3167: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var setPrototypeOf = __webpack_require__(2967); + +// makes subclassing work correct for wrapped built-ins +module.exports = function ($this, dummy, Wrapper) { + var NewTarget, NewTargetPrototype; + if ( + // it can work only with native `setPrototypeOf` + setPrototypeOf && + // we haven't completely correct pre-ES6 way for getting `new.target`, so use this + isCallable(NewTarget = dummy.constructor) && + NewTarget !== Wrapper && + isObject(NewTargetPrototype = NewTarget.prototype) && + NewTargetPrototype !== Wrapper.prototype + ) setPrototypeOf($this, NewTargetPrototype); + return $this; +}; + + +/***/ }), + +/***/ 3706: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var isCallable = __webpack_require__(4901); +var store = __webpack_require__(7629); + +var functionToString = uncurryThis(Function.toString); + +// this helper broken in `core-js@3.4.1-3.4.4`, so we can't use `shared` helper +if (!isCallable(store.inspectSource)) { + store.inspectSource = function (it) { + return functionToString(it); + }; +} + +module.exports = store.inspectSource; + + +/***/ }), + +/***/ 1181: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var NATIVE_WEAK_MAP = __webpack_require__(8622); +var global = __webpack_require__(4475); +var isObject = __webpack_require__(34); +var createNonEnumerableProperty = __webpack_require__(6699); +var hasOwn = __webpack_require__(9297); +var shared = __webpack_require__(7629); +var sharedKey = __webpack_require__(6119); +var hiddenKeys = __webpack_require__(421); + +var OBJECT_ALREADY_INITIALIZED = 'Object already initialized'; +var TypeError = global.TypeError; +var WeakMap = global.WeakMap; +var set, get, has; + +var enforce = function (it) { + return has(it) ? get(it) : set(it, {}); +}; + +var getterFor = function (TYPE) { + return function (it) { + var state; + if (!isObject(it) || (state = get(it)).type !== TYPE) { + throw new TypeError('Incompatible receiver, ' + TYPE + ' required'); + } return state; + }; +}; + +if (NATIVE_WEAK_MAP || shared.state) { + var store = shared.state || (shared.state = new WeakMap()); + /* eslint-disable no-self-assign -- prototype methods protection */ + store.get = store.get; + store.has = store.has; + store.set = store.set; + /* eslint-enable no-self-assign -- prototype methods protection */ + set = function (it, metadata) { + if (store.has(it)) throw new TypeError(OBJECT_ALREADY_INITIALIZED); + metadata.facade = it; + store.set(it, metadata); + return metadata; + }; + get = function (it) { + return store.get(it) || {}; + }; + has = function (it) { + return store.has(it); + }; +} else { + var STATE = sharedKey('state'); + hiddenKeys[STATE] = true; + set = function (it, metadata) { + if (hasOwn(it, STATE)) throw new TypeError(OBJECT_ALREADY_INITIALIZED); + metadata.facade = it; + createNonEnumerableProperty(it, STATE, metadata); + return metadata; + }; + get = function (it) { + return hasOwn(it, STATE) ? it[STATE] : {}; + }; + has = function (it) { + return hasOwn(it, STATE); + }; +} + +module.exports = { + set: set, + get: get, + has: has, + enforce: enforce, + getterFor: getterFor +}; + + +/***/ }), + +/***/ 4209: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var wellKnownSymbol = __webpack_require__(8227); +var Iterators = __webpack_require__(6269); + +var ITERATOR = wellKnownSymbol('iterator'); +var ArrayPrototype = Array.prototype; + +// check on default Array iterator +module.exports = function (it) { + return it !== undefined && (Iterators.Array === it || ArrayPrototype[ITERATOR] === it); +}; + + +/***/ }), + +/***/ 4376: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(4576); + +// `IsArray` abstract operation +// https://tc39.es/ecma262/#sec-isarray +// eslint-disable-next-line es/no-array-isarray -- safe +module.exports = Array.isArray || function isArray(argument) { + return classof(argument) === 'Array'; +}; + + +/***/ }), + +/***/ 1108: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(6955); + +module.exports = function (it) { + var klass = classof(it); + return klass === 'BigInt64Array' || klass === 'BigUint64Array'; +}; + + +/***/ }), + +/***/ 4901: +/***/ ((module) => { + + +// https://tc39.es/ecma262/#sec-IsHTMLDDA-internal-slot +var documentAll = typeof document == 'object' && document.all; + +// `IsCallable` abstract operation +// https://tc39.es/ecma262/#sec-iscallable +// eslint-disable-next-line unicorn/no-typeof-undefined -- required for testing +module.exports = typeof documentAll == 'undefined' && documentAll !== undefined ? function (argument) { + return typeof argument == 'function' || argument === documentAll; +} : function (argument) { + return typeof argument == 'function'; +}; + + +/***/ }), + +/***/ 2796: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); +var isCallable = __webpack_require__(4901); + +var replacement = /#|\.prototype\./; + +var isForced = function (feature, detection) { + var value = data[normalize(feature)]; + return value === POLYFILL ? true + : value === NATIVE ? false + : isCallable(detection) ? fails(detection) + : !!detection; +}; + +var normalize = isForced.normalize = function (string) { + return String(string).replace(replacement, '.').toLowerCase(); +}; + +var data = isForced.data = {}; +var NATIVE = isForced.NATIVE = 'N'; +var POLYFILL = isForced.POLYFILL = 'P'; + +module.exports = isForced; + + +/***/ }), + +/***/ 4117: +/***/ ((module) => { + + +// we can't use just `it == null` since of `document.all` special case +// https://tc39.es/ecma262/#sec-IsHTMLDDA-internal-slot-aec +module.exports = function (it) { + return it === null || it === undefined; +}; + + +/***/ }), + +/***/ 34: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isCallable = __webpack_require__(4901); + +module.exports = function (it) { + return typeof it == 'object' ? it !== null : isCallable(it); +}; + + +/***/ }), + +/***/ 3925: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isObject = __webpack_require__(34); + +module.exports = function (argument) { + return isObject(argument) || argument === null; +}; + + +/***/ }), + +/***/ 6395: +/***/ ((module) => { + + +module.exports = false; + + +/***/ }), + +/***/ 757: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); +var isCallable = __webpack_require__(4901); +var isPrototypeOf = __webpack_require__(1625); +var USE_SYMBOL_AS_UID = __webpack_require__(7040); + +var $Object = Object; + +module.exports = USE_SYMBOL_AS_UID ? function (it) { + return typeof it == 'symbol'; +} : function (it) { + var $Symbol = getBuiltIn('Symbol'); + return isCallable($Symbol) && isPrototypeOf($Symbol.prototype, $Object(it)); +}; + + +/***/ }), + +/***/ 507: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); + +module.exports = function (record, fn, ITERATOR_INSTEAD_OF_RECORD) { + var iterator = ITERATOR_INSTEAD_OF_RECORD ? record : record.iterator; + var next = record.next; + var step, result; + while (!(step = call(next, iterator)).done) { + result = fn(step.value); + if (result !== undefined) return result; + } +}; + + +/***/ }), + +/***/ 2652: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var bind = __webpack_require__(6080); +var call = __webpack_require__(9565); +var anObject = __webpack_require__(8551); +var tryToString = __webpack_require__(6823); +var isArrayIteratorMethod = __webpack_require__(4209); +var lengthOfArrayLike = __webpack_require__(6198); +var isPrototypeOf = __webpack_require__(1625); +var getIterator = __webpack_require__(81); +var getIteratorMethod = __webpack_require__(851); +var iteratorClose = __webpack_require__(9539); + +var $TypeError = TypeError; + +var Result = function (stopped, result) { + this.stopped = stopped; + this.result = result; +}; + +var ResultPrototype = Result.prototype; + +module.exports = function (iterable, unboundFunction, options) { + var that = options && options.that; + var AS_ENTRIES = !!(options && options.AS_ENTRIES); + var IS_RECORD = !!(options && options.IS_RECORD); + var IS_ITERATOR = !!(options && options.IS_ITERATOR); + var INTERRUPTED = !!(options && options.INTERRUPTED); + var fn = bind(unboundFunction, that); + var iterator, iterFn, index, length, result, next, step; + + var stop = function (condition) { + if (iterator) iteratorClose(iterator, 'normal', condition); + return new Result(true, condition); + }; + + var callFn = function (value) { + if (AS_ENTRIES) { + anObject(value); + return INTERRUPTED ? fn(value[0], value[1], stop) : fn(value[0], value[1]); + } return INTERRUPTED ? fn(value, stop) : fn(value); + }; + + if (IS_RECORD) { + iterator = iterable.iterator; + } else if (IS_ITERATOR) { + iterator = iterable; + } else { + iterFn = getIteratorMethod(iterable); + if (!iterFn) throw new $TypeError(tryToString(iterable) + ' is not iterable'); + // optimisation for array iterators + if (isArrayIteratorMethod(iterFn)) { + for (index = 0, length = lengthOfArrayLike(iterable); length > index; index++) { + result = callFn(iterable[index]); + if (result && isPrototypeOf(ResultPrototype, result)) return result; + } return new Result(false); + } + iterator = getIterator(iterable, iterFn); + } + + next = IS_RECORD ? iterable.next : iterator.next; + while (!(step = call(next, iterator)).done) { + try { + result = callFn(step.value); + } catch (error) { + iteratorClose(iterator, 'throw', error); + } + if (typeof result == 'object' && result && isPrototypeOf(ResultPrototype, result)) return result; + } return new Result(false); +}; + + +/***/ }), + +/***/ 9539: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var anObject = __webpack_require__(8551); +var getMethod = __webpack_require__(5966); + +module.exports = function (iterator, kind, value) { + var innerResult, innerError; + anObject(iterator); + try { + innerResult = getMethod(iterator, 'return'); + if (!innerResult) { + if (kind === 'throw') throw value; + return value; + } + innerResult = call(innerResult, iterator); + } catch (error) { + innerError = true; + innerResult = error; + } + if (kind === 'throw') throw value; + if (innerError) throw innerResult; + anObject(innerResult); + return value; +}; + + +/***/ }), + +/***/ 9462: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var create = __webpack_require__(2360); +var createNonEnumerableProperty = __webpack_require__(6699); +var defineBuiltIns = __webpack_require__(6279); +var wellKnownSymbol = __webpack_require__(8227); +var InternalStateModule = __webpack_require__(1181); +var getMethod = __webpack_require__(5966); +var IteratorPrototype = (__webpack_require__(7657).IteratorPrototype); +var createIterResultObject = __webpack_require__(2529); +var iteratorClose = __webpack_require__(9539); + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var ITERATOR_HELPER = 'IteratorHelper'; +var WRAP_FOR_VALID_ITERATOR = 'WrapForValidIterator'; +var setInternalState = InternalStateModule.set; + +var createIteratorProxyPrototype = function (IS_ITERATOR) { + var getInternalState = InternalStateModule.getterFor(IS_ITERATOR ? WRAP_FOR_VALID_ITERATOR : ITERATOR_HELPER); + + return defineBuiltIns(create(IteratorPrototype), { + next: function next() { + var state = getInternalState(this); + // for simplification: + // for `%WrapForValidIteratorPrototype%.next` our `nextHandler` returns `IterResultObject` + // for `%IteratorHelperPrototype%.next` - just a value + if (IS_ITERATOR) return state.nextHandler(); + try { + var result = state.done ? undefined : state.nextHandler(); + return createIterResultObject(result, state.done); + } catch (error) { + state.done = true; + throw error; + } + }, + 'return': function () { + var state = getInternalState(this); + var iterator = state.iterator; + state.done = true; + if (IS_ITERATOR) { + var returnMethod = getMethod(iterator, 'return'); + return returnMethod ? call(returnMethod, iterator) : createIterResultObject(undefined, true); + } + if (state.inner) try { + iteratorClose(state.inner.iterator, 'normal'); + } catch (error) { + return iteratorClose(iterator, 'throw', error); + } + iteratorClose(iterator, 'normal'); + return createIterResultObject(undefined, true); + } + }); +}; + +var WrapForValidIteratorPrototype = createIteratorProxyPrototype(true); +var IteratorHelperPrototype = createIteratorProxyPrototype(false); + +createNonEnumerableProperty(IteratorHelperPrototype, TO_STRING_TAG, 'Iterator Helper'); + +module.exports = function (nextHandler, IS_ITERATOR) { + var IteratorProxy = function Iterator(record, state) { + if (state) { + state.iterator = record.iterator; + state.next = record.next; + } else state = record; + state.type = IS_ITERATOR ? WRAP_FOR_VALID_ITERATOR : ITERATOR_HELPER; + state.nextHandler = nextHandler; + state.counter = 0; + state.done = false; + setInternalState(this, state); + }; + + IteratorProxy.prototype = IS_ITERATOR ? WrapForValidIteratorPrototype : IteratorHelperPrototype; + + return IteratorProxy; +}; + + +/***/ }), + +/***/ 713: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var createIteratorProxy = __webpack_require__(9462); +var callWithSafeIterationClosing = __webpack_require__(6319); + +var IteratorProxy = createIteratorProxy(function () { + var iterator = this.iterator; + var result = anObject(call(this.next, iterator)); + var done = this.done = !!result.done; + if (!done) return callWithSafeIterationClosing(iterator, this.mapper, [result.value, this.counter++], true); +}); + +// `Iterator.prototype.map` method +// https://github.com/tc39/proposal-iterator-helpers +module.exports = function map(mapper) { + anObject(this); + aCallable(mapper); + return new IteratorProxy(getIteratorDirect(this), { + mapper: mapper + }); +}; + + +/***/ }), + +/***/ 7657: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var fails = __webpack_require__(9039); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); +var create = __webpack_require__(2360); +var getPrototypeOf = __webpack_require__(2787); +var defineBuiltIn = __webpack_require__(6840); +var wellKnownSymbol = __webpack_require__(8227); +var IS_PURE = __webpack_require__(6395); + +var ITERATOR = wellKnownSymbol('iterator'); +var BUGGY_SAFARI_ITERATORS = false; + +// `%IteratorPrototype%` object +// https://tc39.es/ecma262/#sec-%iteratorprototype%-object +var IteratorPrototype, PrototypeOfArrayIteratorPrototype, arrayIterator; + +/* eslint-disable es/no-array-prototype-keys -- safe */ +if ([].keys) { + arrayIterator = [].keys(); + // Safari 8 has buggy iterators w/o `next` + if (!('next' in arrayIterator)) BUGGY_SAFARI_ITERATORS = true; + else { + PrototypeOfArrayIteratorPrototype = getPrototypeOf(getPrototypeOf(arrayIterator)); + if (PrototypeOfArrayIteratorPrototype !== Object.prototype) IteratorPrototype = PrototypeOfArrayIteratorPrototype; + } +} + +var NEW_ITERATOR_PROTOTYPE = !isObject(IteratorPrototype) || fails(function () { + var test = {}; + // FF44- legacy iterators case + return IteratorPrototype[ITERATOR].call(test) !== test; +}); + +if (NEW_ITERATOR_PROTOTYPE) IteratorPrototype = {}; +else if (IS_PURE) IteratorPrototype = create(IteratorPrototype); + +// `%IteratorPrototype%[@@iterator]()` method +// https://tc39.es/ecma262/#sec-%iteratorprototype%-@@iterator +if (!isCallable(IteratorPrototype[ITERATOR])) { + defineBuiltIn(IteratorPrototype, ITERATOR, function () { + return this; + }); +} + +module.exports = { + IteratorPrototype: IteratorPrototype, + BUGGY_SAFARI_ITERATORS: BUGGY_SAFARI_ITERATORS +}; + + +/***/ }), + +/***/ 6269: +/***/ ((module) => { + + +module.exports = {}; + + +/***/ }), + +/***/ 6198: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toLength = __webpack_require__(8014); + +// `LengthOfArrayLike` abstract operation +// https://tc39.es/ecma262/#sec-lengthofarraylike +module.exports = function (obj) { + return toLength(obj.length); +}; + + +/***/ }), + +/***/ 283: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var fails = __webpack_require__(9039); +var isCallable = __webpack_require__(4901); +var hasOwn = __webpack_require__(9297); +var DESCRIPTORS = __webpack_require__(3724); +var CONFIGURABLE_FUNCTION_NAME = (__webpack_require__(350).CONFIGURABLE); +var inspectSource = __webpack_require__(3706); +var InternalStateModule = __webpack_require__(1181); + +var enforceInternalState = InternalStateModule.enforce; +var getInternalState = InternalStateModule.get; +var $String = String; +// eslint-disable-next-line es/no-object-defineproperty -- safe +var defineProperty = Object.defineProperty; +var stringSlice = uncurryThis(''.slice); +var replace = uncurryThis(''.replace); +var join = uncurryThis([].join); + +var CONFIGURABLE_LENGTH = DESCRIPTORS && !fails(function () { + return defineProperty(function () { /* empty */ }, 'length', { value: 8 }).length !== 8; +}); + +var TEMPLATE = String(String).split('String'); + +var makeBuiltIn = module.exports = function (value, name, options) { + if (stringSlice($String(name), 0, 7) === 'Symbol(') { + name = '[' + replace($String(name), /^Symbol\(([^)]*)\).*$/, '$1') + ']'; + } + if (options && options.getter) name = 'get ' + name; + if (options && options.setter) name = 'set ' + name; + if (!hasOwn(value, 'name') || (CONFIGURABLE_FUNCTION_NAME && value.name !== name)) { + if (DESCRIPTORS) defineProperty(value, 'name', { value: name, configurable: true }); + else value.name = name; + } + if (CONFIGURABLE_LENGTH && options && hasOwn(options, 'arity') && value.length !== options.arity) { + defineProperty(value, 'length', { value: options.arity }); + } + try { + if (options && hasOwn(options, 'constructor') && options.constructor) { + if (DESCRIPTORS) defineProperty(value, 'prototype', { writable: false }); + // in V8 ~ Chrome 53, prototypes of some methods, like `Array.prototype.values`, are non-writable + } else if (value.prototype) value.prototype = undefined; + } catch (error) { /* empty */ } + var state = enforceInternalState(value); + if (!hasOwn(state, 'source')) { + state.source = join(TEMPLATE, typeof name == 'string' ? name : ''); + } return value; +}; + +// add fake Function#toString for correct work wrapped methods / constructors with methods like LoDash isNative +// eslint-disable-next-line no-extend-native -- required +Function.prototype.toString = makeBuiltIn(function toString() { + return isCallable(this) && getInternalState(this).source || inspectSource(this); +}, 'toString'); + + +/***/ }), + +/***/ 741: +/***/ ((module) => { + + +var ceil = Math.ceil; +var floor = Math.floor; + +// `Math.trunc` method +// https://tc39.es/ecma262/#sec-math.trunc +// eslint-disable-next-line es/no-math-trunc -- safe +module.exports = Math.trunc || function trunc(x) { + var n = +x; + return (n > 0 ? floor : ceil)(n); +}; + + +/***/ }), + +/***/ 6043: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aCallable = __webpack_require__(9306); + +var $TypeError = TypeError; + +var PromiseCapability = function (C) { + var resolve, reject; + this.promise = new C(function ($$resolve, $$reject) { + if (resolve !== undefined || reject !== undefined) throw new $TypeError('Bad Promise constructor'); + resolve = $$resolve; + reject = $$reject; + }); + this.resolve = aCallable(resolve); + this.reject = aCallable(reject); +}; + +// `NewPromiseCapability` abstract operation +// https://tc39.es/ecma262/#sec-newpromisecapability +module.exports.f = function (C) { + return new PromiseCapability(C); +}; + + +/***/ }), + +/***/ 2603: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toString = __webpack_require__(655); + +module.exports = function (argument, $default) { + return argument === undefined ? arguments.length < 2 ? '' : $default : toString(argument); +}; + + +/***/ }), + +/***/ 2360: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* global ActiveXObject -- old IE, WSH */ +var anObject = __webpack_require__(8551); +var definePropertiesModule = __webpack_require__(6801); +var enumBugKeys = __webpack_require__(8727); +var hiddenKeys = __webpack_require__(421); +var html = __webpack_require__(397); +var documentCreateElement = __webpack_require__(4055); +var sharedKey = __webpack_require__(6119); + +var GT = '>'; +var LT = '<'; +var PROTOTYPE = 'prototype'; +var SCRIPT = 'script'; +var IE_PROTO = sharedKey('IE_PROTO'); + +var EmptyConstructor = function () { /* empty */ }; + +var scriptTag = function (content) { + return LT + SCRIPT + GT + content + LT + '/' + SCRIPT + GT; +}; + +// Create object with fake `null` prototype: use ActiveX Object with cleared prototype +var NullProtoObjectViaActiveX = function (activeXDocument) { + activeXDocument.write(scriptTag('')); + activeXDocument.close(); + var temp = activeXDocument.parentWindow.Object; + activeXDocument = null; // avoid memory leak + return temp; +}; + +// Create object with fake `null` prototype: use iframe Object with cleared prototype +var NullProtoObjectViaIFrame = function () { + // Thrash, waste and sodomy: IE GC bug + var iframe = documentCreateElement('iframe'); + var JS = 'java' + SCRIPT + ':'; + var iframeDocument; + iframe.style.display = 'none'; + html.appendChild(iframe); + // https://github.com/zloirock/core-js/issues/475 + iframe.src = String(JS); + iframeDocument = iframe.contentWindow.document; + iframeDocument.open(); + iframeDocument.write(scriptTag('document.F=Object')); + iframeDocument.close(); + return iframeDocument.F; +}; + +// Check for document.domain and active x support +// No need to use active x approach when document.domain is not set +// see https://github.com/es-shims/es5-shim/issues/150 +// variation of https://github.com/kitcambridge/es5-shim/commit/4f738ac066346 +// avoid IE GC bug +var activeXDocument; +var NullProtoObject = function () { + try { + activeXDocument = new ActiveXObject('htmlfile'); + } catch (error) { /* ignore */ } + NullProtoObject = typeof document != 'undefined' + ? document.domain && activeXDocument + ? NullProtoObjectViaActiveX(activeXDocument) // old IE + : NullProtoObjectViaIFrame() + : NullProtoObjectViaActiveX(activeXDocument); // WSH + var length = enumBugKeys.length; + while (length--) delete NullProtoObject[PROTOTYPE][enumBugKeys[length]]; + return NullProtoObject(); +}; + +hiddenKeys[IE_PROTO] = true; + +// `Object.create` method +// https://tc39.es/ecma262/#sec-object.create +// eslint-disable-next-line es/no-object-create -- safe +module.exports = Object.create || function create(O, Properties) { + var result; + if (O !== null) { + EmptyConstructor[PROTOTYPE] = anObject(O); + result = new EmptyConstructor(); + EmptyConstructor[PROTOTYPE] = null; + // add "__proto__" for Object.getPrototypeOf polyfill + result[IE_PROTO] = O; + } else result = NullProtoObject(); + return Properties === undefined ? result : definePropertiesModule.f(result, Properties); +}; + + +/***/ }), + +/***/ 6801: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var V8_PROTOTYPE_DEFINE_BUG = __webpack_require__(8686); +var definePropertyModule = __webpack_require__(4913); +var anObject = __webpack_require__(8551); +var toIndexedObject = __webpack_require__(5397); +var objectKeys = __webpack_require__(1072); + +// `Object.defineProperties` method +// https://tc39.es/ecma262/#sec-object.defineproperties +// eslint-disable-next-line es/no-object-defineproperties -- safe +exports.f = DESCRIPTORS && !V8_PROTOTYPE_DEFINE_BUG ? Object.defineProperties : function defineProperties(O, Properties) { + anObject(O); + var props = toIndexedObject(Properties); + var keys = objectKeys(Properties); + var length = keys.length; + var index = 0; + var key; + while (length > index) definePropertyModule.f(O, key = keys[index++], props[key]); + return O; +}; + + +/***/ }), + +/***/ 4913: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var IE8_DOM_DEFINE = __webpack_require__(5917); +var V8_PROTOTYPE_DEFINE_BUG = __webpack_require__(8686); +var anObject = __webpack_require__(8551); +var toPropertyKey = __webpack_require__(6969); + +var $TypeError = TypeError; +// eslint-disable-next-line es/no-object-defineproperty -- safe +var $defineProperty = Object.defineProperty; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var $getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; +var ENUMERABLE = 'enumerable'; +var CONFIGURABLE = 'configurable'; +var WRITABLE = 'writable'; + +// `Object.defineProperty` method +// https://tc39.es/ecma262/#sec-object.defineproperty +exports.f = DESCRIPTORS ? V8_PROTOTYPE_DEFINE_BUG ? function defineProperty(O, P, Attributes) { + anObject(O); + P = toPropertyKey(P); + anObject(Attributes); + if (typeof O === 'function' && P === 'prototype' && 'value' in Attributes && WRITABLE in Attributes && !Attributes[WRITABLE]) { + var current = $getOwnPropertyDescriptor(O, P); + if (current && current[WRITABLE]) { + O[P] = Attributes.value; + Attributes = { + configurable: CONFIGURABLE in Attributes ? Attributes[CONFIGURABLE] : current[CONFIGURABLE], + enumerable: ENUMERABLE in Attributes ? Attributes[ENUMERABLE] : current[ENUMERABLE], + writable: false + }; + } + } return $defineProperty(O, P, Attributes); +} : $defineProperty : function defineProperty(O, P, Attributes) { + anObject(O); + P = toPropertyKey(P); + anObject(Attributes); + if (IE8_DOM_DEFINE) try { + return $defineProperty(O, P, Attributes); + } catch (error) { /* empty */ } + if ('get' in Attributes || 'set' in Attributes) throw new $TypeError('Accessors not supported'); + if ('value' in Attributes) O[P] = Attributes.value; + return O; +}; + + +/***/ }), + +/***/ 7347: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var call = __webpack_require__(9565); +var propertyIsEnumerableModule = __webpack_require__(8773); +var createPropertyDescriptor = __webpack_require__(6980); +var toIndexedObject = __webpack_require__(5397); +var toPropertyKey = __webpack_require__(6969); +var hasOwn = __webpack_require__(9297); +var IE8_DOM_DEFINE = __webpack_require__(5917); + +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var $getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; + +// `Object.getOwnPropertyDescriptor` method +// https://tc39.es/ecma262/#sec-object.getownpropertydescriptor +exports.f = DESCRIPTORS ? $getOwnPropertyDescriptor : function getOwnPropertyDescriptor(O, P) { + O = toIndexedObject(O); + P = toPropertyKey(P); + if (IE8_DOM_DEFINE) try { + return $getOwnPropertyDescriptor(O, P); + } catch (error) { /* empty */ } + if (hasOwn(O, P)) return createPropertyDescriptor(!call(propertyIsEnumerableModule.f, O, P), O[P]); +}; + + +/***/ }), + +/***/ 8480: +/***/ ((__unused_webpack_module, exports, __webpack_require__) => { + + +var internalObjectKeys = __webpack_require__(1828); +var enumBugKeys = __webpack_require__(8727); + +var hiddenKeys = enumBugKeys.concat('length', 'prototype'); + +// `Object.getOwnPropertyNames` method +// https://tc39.es/ecma262/#sec-object.getownpropertynames +// eslint-disable-next-line es/no-object-getownpropertynames -- safe +exports.f = Object.getOwnPropertyNames || function getOwnPropertyNames(O) { + return internalObjectKeys(O, hiddenKeys); +}; + + +/***/ }), + +/***/ 3717: +/***/ ((__unused_webpack_module, exports) => { + + +// eslint-disable-next-line es/no-object-getownpropertysymbols -- safe +exports.f = Object.getOwnPropertySymbols; + + +/***/ }), + +/***/ 2787: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var hasOwn = __webpack_require__(9297); +var isCallable = __webpack_require__(4901); +var toObject = __webpack_require__(8981); +var sharedKey = __webpack_require__(6119); +var CORRECT_PROTOTYPE_GETTER = __webpack_require__(2211); + +var IE_PROTO = sharedKey('IE_PROTO'); +var $Object = Object; +var ObjectPrototype = $Object.prototype; + +// `Object.getPrototypeOf` method +// https://tc39.es/ecma262/#sec-object.getprototypeof +// eslint-disable-next-line es/no-object-getprototypeof -- safe +module.exports = CORRECT_PROTOTYPE_GETTER ? $Object.getPrototypeOf : function (O) { + var object = toObject(O); + if (hasOwn(object, IE_PROTO)) return object[IE_PROTO]; + var constructor = object.constructor; + if (isCallable(constructor) && object instanceof constructor) { + return constructor.prototype; + } return object instanceof $Object ? ObjectPrototype : null; +}; + + +/***/ }), + +/***/ 1625: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +module.exports = uncurryThis({}.isPrototypeOf); + + +/***/ }), + +/***/ 1828: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var hasOwn = __webpack_require__(9297); +var toIndexedObject = __webpack_require__(5397); +var indexOf = (__webpack_require__(9617).indexOf); +var hiddenKeys = __webpack_require__(421); + +var push = uncurryThis([].push); + +module.exports = function (object, names) { + var O = toIndexedObject(object); + var i = 0; + var result = []; + var key; + for (key in O) !hasOwn(hiddenKeys, key) && hasOwn(O, key) && push(result, key); + // Don't enum bug & hidden keys + while (names.length > i) if (hasOwn(O, key = names[i++])) { + ~indexOf(result, key) || push(result, key); + } + return result; +}; + + +/***/ }), + +/***/ 1072: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var internalObjectKeys = __webpack_require__(1828); +var enumBugKeys = __webpack_require__(8727); + +// `Object.keys` method +// https://tc39.es/ecma262/#sec-object.keys +// eslint-disable-next-line es/no-object-keys -- safe +module.exports = Object.keys || function keys(O) { + return internalObjectKeys(O, enumBugKeys); +}; + + +/***/ }), + +/***/ 8773: +/***/ ((__unused_webpack_module, exports) => { + + +var $propertyIsEnumerable = {}.propertyIsEnumerable; +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; + +// Nashorn ~ JDK8 bug +var NASHORN_BUG = getOwnPropertyDescriptor && !$propertyIsEnumerable.call({ 1: 2 }, 1); + +// `Object.prototype.propertyIsEnumerable` method implementation +// https://tc39.es/ecma262/#sec-object.prototype.propertyisenumerable +exports.f = NASHORN_BUG ? function propertyIsEnumerable(V) { + var descriptor = getOwnPropertyDescriptor(this, V); + return !!descriptor && descriptor.enumerable; +} : $propertyIsEnumerable; + + +/***/ }), + +/***/ 2967: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* eslint-disable no-proto -- safe */ +var uncurryThisAccessor = __webpack_require__(6706); +var isObject = __webpack_require__(34); +var requireObjectCoercible = __webpack_require__(7750); +var aPossiblePrototype = __webpack_require__(3506); + +// `Object.setPrototypeOf` method +// https://tc39.es/ecma262/#sec-object.setprototypeof +// Works with __proto__ only. Old v8 can't work with null proto objects. +// eslint-disable-next-line es/no-object-setprototypeof -- safe +module.exports = Object.setPrototypeOf || ('__proto__' in {} ? function () { + var CORRECT_SETTER = false; + var test = {}; + var setter; + try { + setter = uncurryThisAccessor(Object.prototype, '__proto__', 'set'); + setter(test, []); + CORRECT_SETTER = test instanceof Array; + } catch (error) { /* empty */ } + return function setPrototypeOf(O, proto) { + requireObjectCoercible(O); + aPossiblePrototype(proto); + if (!isObject(O)) return O; + if (CORRECT_SETTER) setter(O, proto); + else O.__proto__ = proto; + return O; + }; +}() : undefined); + + +/***/ }), + +/***/ 4270: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var isCallable = __webpack_require__(4901); +var isObject = __webpack_require__(34); + +var $TypeError = TypeError; + +// `OrdinaryToPrimitive` abstract operation +// https://tc39.es/ecma262/#sec-ordinarytoprimitive +module.exports = function (input, pref) { + var fn, val; + if (pref === 'string' && isCallable(fn = input.toString) && !isObject(val = call(fn, input))) return val; + if (isCallable(fn = input.valueOf) && !isObject(val = call(fn, input))) return val; + if (pref !== 'string' && isCallable(fn = input.toString) && !isObject(val = call(fn, input))) return val; + throw new $TypeError("Can't convert object to primitive value"); +}; + + +/***/ }), + +/***/ 5031: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); +var uncurryThis = __webpack_require__(9504); +var getOwnPropertyNamesModule = __webpack_require__(8480); +var getOwnPropertySymbolsModule = __webpack_require__(3717); +var anObject = __webpack_require__(8551); + +var concat = uncurryThis([].concat); + +// all object keys, includes non-enumerable and symbols +module.exports = getBuiltIn('Reflect', 'ownKeys') || function ownKeys(it) { + var keys = getOwnPropertyNamesModule.f(anObject(it)); + var getOwnPropertySymbols = getOwnPropertySymbolsModule.f; + return getOwnPropertySymbols ? concat(keys, getOwnPropertySymbols(it)) : keys; +}; + + +/***/ }), + +/***/ 7979: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var anObject = __webpack_require__(8551); + +// `RegExp.prototype.flags` getter implementation +// https://tc39.es/ecma262/#sec-get-regexp.prototype.flags +module.exports = function () { + var that = anObject(this); + var result = ''; + if (that.hasIndices) result += 'd'; + if (that.global) result += 'g'; + if (that.ignoreCase) result += 'i'; + if (that.multiline) result += 'm'; + if (that.dotAll) result += 's'; + if (that.unicode) result += 'u'; + if (that.unicodeSets) result += 'v'; + if (that.sticky) result += 'y'; + return result; +}; + + +/***/ }), + +/***/ 7750: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var isNullOrUndefined = __webpack_require__(4117); + +var $TypeError = TypeError; + +// `RequireObjectCoercible` abstract operation +// https://tc39.es/ecma262/#sec-requireobjectcoercible +module.exports = function (it) { + if (isNullOrUndefined(it)) throw new $TypeError("Can't call method on " + it); + return it; +}; + + +/***/ }), + +/***/ 9286: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var SetHelpers = __webpack_require__(4402); +var iterate = __webpack_require__(8469); + +var Set = SetHelpers.Set; +var add = SetHelpers.add; + +module.exports = function (set) { + var result = new Set(); + iterate(set, function (it) { + add(result, it); + }); + return result; +}; + + +/***/ }), + +/***/ 3440: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var SetHelpers = __webpack_require__(4402); +var clone = __webpack_require__(9286); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSet = __webpack_require__(8469); +var iterateSimple = __webpack_require__(507); + +var has = SetHelpers.has; +var remove = SetHelpers.remove; + +// `Set.prototype.difference` method +// https://github.com/tc39/proposal-set-methods +module.exports = function difference(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + var result = clone(O); + if (size(O) <= otherRec.size) iterateSet(O, function (e) { + if (otherRec.includes(e)) remove(result, e); + }); + else iterateSimple(otherRec.getIterator(), function (e) { + if (has(O, e)) remove(result, e); + }); + return result; +}; + + +/***/ }), + +/***/ 4402: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +// eslint-disable-next-line es/no-set -- safe +var SetPrototype = Set.prototype; + +module.exports = { + // eslint-disable-next-line es/no-set -- safe + Set: Set, + add: uncurryThis(SetPrototype.add), + has: uncurryThis(SetPrototype.has), + remove: uncurryThis(SetPrototype['delete']), + proto: SetPrototype +}; + + +/***/ }), + +/***/ 8750: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var SetHelpers = __webpack_require__(4402); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSet = __webpack_require__(8469); +var iterateSimple = __webpack_require__(507); + +var Set = SetHelpers.Set; +var add = SetHelpers.add; +var has = SetHelpers.has; + +// `Set.prototype.intersection` method +// https://github.com/tc39/proposal-set-methods +module.exports = function intersection(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + var result = new Set(); + + if (size(O) > otherRec.size) { + iterateSimple(otherRec.getIterator(), function (e) { + if (has(O, e)) add(result, e); + }); + } else { + iterateSet(O, function (e) { + if (otherRec.includes(e)) add(result, e); + }); + } + + return result; +}; + + +/***/ }), + +/***/ 4449: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var has = (__webpack_require__(4402).has); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSet = __webpack_require__(8469); +var iterateSimple = __webpack_require__(507); +var iteratorClose = __webpack_require__(9539); + +// `Set.prototype.isDisjointFrom` method +// https://tc39.github.io/proposal-set-methods/#Set.prototype.isDisjointFrom +module.exports = function isDisjointFrom(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + if (size(O) <= otherRec.size) return iterateSet(O, function (e) { + if (otherRec.includes(e)) return false; + }, true) !== false; + var iterator = otherRec.getIterator(); + return iterateSimple(iterator, function (e) { + if (has(O, e)) return iteratorClose(iterator, 'normal', false); + }) !== false; +}; + + +/***/ }), + +/***/ 3838: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var size = __webpack_require__(5170); +var iterate = __webpack_require__(8469); +var getSetRecord = __webpack_require__(3789); + +// `Set.prototype.isSubsetOf` method +// https://tc39.github.io/proposal-set-methods/#Set.prototype.isSubsetOf +module.exports = function isSubsetOf(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + if (size(O) > otherRec.size) return false; + return iterate(O, function (e) { + if (!otherRec.includes(e)) return false; + }, true) !== false; +}; + + +/***/ }), + +/***/ 8527: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var has = (__webpack_require__(4402).has); +var size = __webpack_require__(5170); +var getSetRecord = __webpack_require__(3789); +var iterateSimple = __webpack_require__(507); +var iteratorClose = __webpack_require__(9539); + +// `Set.prototype.isSupersetOf` method +// https://tc39.github.io/proposal-set-methods/#Set.prototype.isSupersetOf +module.exports = function isSupersetOf(other) { + var O = aSet(this); + var otherRec = getSetRecord(other); + if (size(O) < otherRec.size) return false; + var iterator = otherRec.getIterator(); + return iterateSimple(iterator, function (e) { + if (!has(O, e)) return iteratorClose(iterator, 'normal', false); + }) !== false; +}; + + +/***/ }), + +/***/ 8469: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); +var iterateSimple = __webpack_require__(507); +var SetHelpers = __webpack_require__(4402); + +var Set = SetHelpers.Set; +var SetPrototype = SetHelpers.proto; +var forEach = uncurryThis(SetPrototype.forEach); +var keys = uncurryThis(SetPrototype.keys); +var next = keys(new Set()).next; + +module.exports = function (set, fn, interruptible) { + return interruptible ? iterateSimple({ iterator: keys(set), next: next }, fn) : forEach(set, fn); +}; + + +/***/ }), + +/***/ 4916: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var getBuiltIn = __webpack_require__(7751); + +var createSetLike = function (size) { + return { + size: size, + has: function () { + return false; + }, + keys: function () { + return { + next: function () { + return { done: true }; + } + }; + } + }; +}; + +module.exports = function (name) { + var Set = getBuiltIn('Set'); + try { + new Set()[name](createSetLike(0)); + try { + // late spec change, early WebKit ~ Safari 17.0 beta implementation does not pass it + // https://github.com/tc39/proposal-set-methods/pull/88 + new Set()[name](createSetLike(-1)); + return false; + } catch (error2) { + return true; + } + } catch (error) { + return false; + } +}; + + +/***/ }), + +/***/ 5170: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThisAccessor = __webpack_require__(6706); +var SetHelpers = __webpack_require__(4402); + +module.exports = uncurryThisAccessor(SetHelpers.proto, 'size', 'get') || function (set) { + return set.size; +}; + + +/***/ }), + +/***/ 3650: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var SetHelpers = __webpack_require__(4402); +var clone = __webpack_require__(9286); +var getSetRecord = __webpack_require__(3789); +var iterateSimple = __webpack_require__(507); + +var add = SetHelpers.add; +var has = SetHelpers.has; +var remove = SetHelpers.remove; + +// `Set.prototype.symmetricDifference` method +// https://github.com/tc39/proposal-set-methods +module.exports = function symmetricDifference(other) { + var O = aSet(this); + var keysIter = getSetRecord(other).getIterator(); + var result = clone(O); + iterateSimple(keysIter, function (e) { + if (has(O, e)) remove(result, e); + else add(result, e); + }); + return result; +}; + + +/***/ }), + +/***/ 4204: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var aSet = __webpack_require__(7080); +var add = (__webpack_require__(4402).add); +var clone = __webpack_require__(9286); +var getSetRecord = __webpack_require__(3789); +var iterateSimple = __webpack_require__(507); + +// `Set.prototype.union` method +// https://github.com/tc39/proposal-set-methods +module.exports = function union(other) { + var O = aSet(this); + var keysIter = getSetRecord(other).getIterator(); + var result = clone(O); + iterateSimple(keysIter, function (it) { + add(result, it); + }); + return result; +}; + + +/***/ }), + +/***/ 6119: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var shared = __webpack_require__(5745); +var uid = __webpack_require__(3392); + +var keys = shared('keys'); + +module.exports = function (key) { + return keys[key] || (keys[key] = uid(key)); +}; + + +/***/ }), + +/***/ 7629: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var IS_PURE = __webpack_require__(6395); +var globalThis = __webpack_require__(4475); +var defineGlobalProperty = __webpack_require__(9433); + +var SHARED = '__core-js_shared__'; +var store = module.exports = globalThis[SHARED] || defineGlobalProperty(SHARED, {}); + +(store.versions || (store.versions = [])).push({ + version: '3.37.1', + mode: IS_PURE ? 'pure' : 'global', + copyright: '© 2014-2024 Denis Pushkarev (zloirock.ru)', + license: 'https://github.com/zloirock/core-js/blob/v3.37.1/LICENSE', + source: 'https://github.com/zloirock/core-js' +}); + + +/***/ }), + +/***/ 5745: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var store = __webpack_require__(7629); + +module.exports = function (key, value) { + return store[key] || (store[key] = value || {}); +}; + + +/***/ }), + +/***/ 1548: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var fails = __webpack_require__(9039); +var V8 = __webpack_require__(7388); +var IS_BROWSER = __webpack_require__(7290); +var IS_DENO = __webpack_require__(516); +var IS_NODE = __webpack_require__(9088); + +var structuredClone = global.structuredClone; + +module.exports = !!structuredClone && !fails(function () { + // prevent V8 ArrayBufferDetaching protector cell invalidation and performance degradation + // https://github.com/zloirock/core-js/issues/679 + if ((IS_DENO && V8 > 92) || (IS_NODE && V8 > 94) || (IS_BROWSER && V8 > 97)) return false; + var buffer = new ArrayBuffer(8); + var clone = structuredClone(buffer, { transfer: [buffer] }); + return buffer.byteLength !== 0 || clone.byteLength !== 8; +}); + + +/***/ }), + +/***/ 4495: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* eslint-disable es/no-symbol -- required for testing */ +var V8_VERSION = __webpack_require__(7388); +var fails = __webpack_require__(9039); +var global = __webpack_require__(4475); + +var $String = global.String; + +// eslint-disable-next-line es/no-object-getownpropertysymbols -- required for testing +module.exports = !!Object.getOwnPropertySymbols && !fails(function () { + var symbol = Symbol('symbol detection'); + // Chrome 38 Symbol has incorrect toString conversion + // `get-own-property-symbols` polyfill symbols converted to object are not Symbol instances + // nb: Do not call `String` directly to avoid this being optimized out to `symbol+''` which will, + // of course, fail. + return !$String(symbol) || !(Object(symbol) instanceof Symbol) || + // Chrome 38-40 symbols are not inherited from DOM collections prototypes to instances + !Symbol.sham && V8_VERSION && V8_VERSION < 41; +}); + + +/***/ }), + +/***/ 5610: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); + +var max = Math.max; +var min = Math.min; + +// Helper for a popular repeating case of the spec: +// Let integer be ? ToInteger(index). +// If integer < 0, let result be max((length + integer), 0); else let result be min(integer, length). +module.exports = function (index, length) { + var integer = toIntegerOrInfinity(index); + return integer < 0 ? max(integer + length, 0) : min(integer, length); +}; + + +/***/ }), + +/***/ 5854: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toPrimitive = __webpack_require__(2777); + +var $TypeError = TypeError; + +// `ToBigInt` abstract operation +// https://tc39.es/ecma262/#sec-tobigint +module.exports = function (argument) { + var prim = toPrimitive(argument, 'number'); + if (typeof prim == 'number') throw new $TypeError("Can't convert number to bigint"); + // eslint-disable-next-line es/no-bigint -- safe + return BigInt(prim); +}; + + +/***/ }), + +/***/ 7696: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); +var toLength = __webpack_require__(8014); + +var $RangeError = RangeError; + +// `ToIndex` abstract operation +// https://tc39.es/ecma262/#sec-toindex +module.exports = function (it) { + if (it === undefined) return 0; + var number = toIntegerOrInfinity(it); + var length = toLength(number); + if (number !== length) throw new $RangeError('Wrong length or index'); + return length; +}; + + +/***/ }), + +/***/ 5397: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +// toObject with fallback for non-array-like ES3 strings +var IndexedObject = __webpack_require__(7055); +var requireObjectCoercible = __webpack_require__(7750); + +module.exports = function (it) { + return IndexedObject(requireObjectCoercible(it)); +}; + + +/***/ }), + +/***/ 1291: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var trunc = __webpack_require__(741); + +// `ToIntegerOrInfinity` abstract operation +// https://tc39.es/ecma262/#sec-tointegerorinfinity +module.exports = function (argument) { + var number = +argument; + // eslint-disable-next-line no-self-compare -- NaN check + return number !== number || number === 0 ? 0 : trunc(number); +}; + + +/***/ }), + +/***/ 8014: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toIntegerOrInfinity = __webpack_require__(1291); + +var min = Math.min; + +// `ToLength` abstract operation +// https://tc39.es/ecma262/#sec-tolength +module.exports = function (argument) { + var len = toIntegerOrInfinity(argument); + return len > 0 ? min(len, 0x1FFFFFFFFFFFFF) : 0; // 2 ** 53 - 1 == 9007199254740991 +}; + + +/***/ }), + +/***/ 8981: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var requireObjectCoercible = __webpack_require__(7750); + +var $Object = Object; + +// `ToObject` abstract operation +// https://tc39.es/ecma262/#sec-toobject +module.exports = function (argument) { + return $Object(requireObjectCoercible(argument)); +}; + + +/***/ }), + +/***/ 2777: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var call = __webpack_require__(9565); +var isObject = __webpack_require__(34); +var isSymbol = __webpack_require__(757); +var getMethod = __webpack_require__(5966); +var ordinaryToPrimitive = __webpack_require__(4270); +var wellKnownSymbol = __webpack_require__(8227); + +var $TypeError = TypeError; +var TO_PRIMITIVE = wellKnownSymbol('toPrimitive'); + +// `ToPrimitive` abstract operation +// https://tc39.es/ecma262/#sec-toprimitive +module.exports = function (input, pref) { + if (!isObject(input) || isSymbol(input)) return input; + var exoticToPrim = getMethod(input, TO_PRIMITIVE); + var result; + if (exoticToPrim) { + if (pref === undefined) pref = 'default'; + result = call(exoticToPrim, input, pref); + if (!isObject(result) || isSymbol(result)) return result; + throw new $TypeError("Can't convert object to primitive value"); + } + if (pref === undefined) pref = 'number'; + return ordinaryToPrimitive(input, pref); +}; + + +/***/ }), + +/***/ 6969: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var toPrimitive = __webpack_require__(2777); +var isSymbol = __webpack_require__(757); + +// `ToPropertyKey` abstract operation +// https://tc39.es/ecma262/#sec-topropertykey +module.exports = function (argument) { + var key = toPrimitive(argument, 'string'); + return isSymbol(key) ? key : key + ''; +}; + + +/***/ }), + +/***/ 2140: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var wellKnownSymbol = __webpack_require__(8227); + +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); +var test = {}; + +test[TO_STRING_TAG] = 'z'; + +module.exports = String(test) === '[object z]'; + + +/***/ }), + +/***/ 655: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var classof = __webpack_require__(6955); + +var $String = String; + +module.exports = function (argument) { + if (classof(argument) === 'Symbol') throw new TypeError('Cannot convert a Symbol value to a string'); + return $String(argument); +}; + + +/***/ }), + +/***/ 9714: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var IS_NODE = __webpack_require__(9088); + +module.exports = function (name) { + try { + // eslint-disable-next-line no-new-func -- safe + if (IS_NODE) return Function('return require("' + name + '")')(); + } catch (error) { /* empty */ } +}; + + +/***/ }), + +/***/ 6823: +/***/ ((module) => { + + +var $String = String; + +module.exports = function (argument) { + try { + return $String(argument); + } catch (error) { + return 'Object'; + } +}; + + +/***/ }), + +/***/ 3392: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var uncurryThis = __webpack_require__(9504); + +var id = 0; +var postfix = Math.random(); +var toString = uncurryThis(1.0.toString); + +module.exports = function (key) { + return 'Symbol(' + (key === undefined ? '' : key) + ')_' + toString(++id + postfix, 36); +}; + + +/***/ }), + +/***/ 7040: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +/* eslint-disable es/no-symbol -- required for testing */ +var NATIVE_SYMBOL = __webpack_require__(4495); + +module.exports = NATIVE_SYMBOL + && !Symbol.sham + && typeof Symbol.iterator == 'symbol'; + + +/***/ }), + +/***/ 8686: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var fails = __webpack_require__(9039); + +// V8 ~ Chrome 36- +// https://bugs.chromium.org/p/v8/issues/detail?id=3334 +module.exports = DESCRIPTORS && fails(function () { + // eslint-disable-next-line es/no-object-defineproperty -- required for testing + return Object.defineProperty(function () { /* empty */ }, 'prototype', { + value: 42, + writable: false + }).prototype !== 42; +}); + + +/***/ }), + +/***/ 2812: +/***/ ((module) => { + + +var $TypeError = TypeError; + +module.exports = function (passed, required) { + if (passed < required) throw new $TypeError('Not enough arguments'); + return passed; +}; + + +/***/ }), + +/***/ 8622: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var isCallable = __webpack_require__(4901); + +var WeakMap = global.WeakMap; + +module.exports = isCallable(WeakMap) && /native code/.test(String(WeakMap)); + + +/***/ }), + +/***/ 8227: +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var shared = __webpack_require__(5745); +var hasOwn = __webpack_require__(9297); +var uid = __webpack_require__(3392); +var NATIVE_SYMBOL = __webpack_require__(4495); +var USE_SYMBOL_AS_UID = __webpack_require__(7040); + +var Symbol = global.Symbol; +var WellKnownSymbolsStore = shared('wks'); +var createWellKnownSymbol = USE_SYMBOL_AS_UID ? Symbol['for'] || Symbol : Symbol && Symbol.withoutSetter || uid; + +module.exports = function (name) { + if (!hasOwn(WellKnownSymbolsStore, name)) { + WellKnownSymbolsStore[name] = NATIVE_SYMBOL && hasOwn(Symbol, name) + ? Symbol[name] + : createWellKnownSymbol('Symbol.' + name); + } return WellKnownSymbolsStore[name]; +}; + + +/***/ }), + +/***/ 6573: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var defineBuiltInAccessor = __webpack_require__(2106); +var isDetached = __webpack_require__(3238); + +var ArrayBufferPrototype = ArrayBuffer.prototype; + +if (DESCRIPTORS && !('detached' in ArrayBufferPrototype)) { + defineBuiltInAccessor(ArrayBufferPrototype, 'detached', { + configurable: true, + get: function detached() { + return isDetached(this); + } + }); +} + + +/***/ }), + +/***/ 7936: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var $transfer = __webpack_require__(5636); + +// `ArrayBuffer.prototype.transferToFixedLength` method +// https://tc39.es/proposal-arraybuffer-transfer/#sec-arraybuffer.prototype.transfertofixedlength +if ($transfer) $({ target: 'ArrayBuffer', proto: true }, { + transferToFixedLength: function transferToFixedLength() { + return $transfer(this, arguments.length ? arguments[0] : undefined, false); + } +}); + + +/***/ }), + +/***/ 8100: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var $transfer = __webpack_require__(5636); + +// `ArrayBuffer.prototype.transfer` method +// https://tc39.es/proposal-arraybuffer-transfer/#sec-arraybuffer.prototype.transfer +if ($transfer) $({ target: 'ArrayBuffer', proto: true }, { + transfer: function transfer() { + return $transfer(this, arguments.length ? arguments[0] : undefined, true); + } +}); + + +/***/ }), + +/***/ 4114: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var toObject = __webpack_require__(8981); +var lengthOfArrayLike = __webpack_require__(6198); +var setArrayLength = __webpack_require__(4527); +var doesNotExceedSafeInteger = __webpack_require__(6837); +var fails = __webpack_require__(9039); + +var INCORRECT_TO_LENGTH = fails(function () { + return [].push.call({ length: 0x100000000 }, 1) !== 4294967297; +}); + +// V8 <= 121 and Safari <= 15.4; FF < 23 throws InternalError +// https://bugs.chromium.org/p/v8/issues/detail?id=12681 +var properErrorOnNonWritableLength = function () { + try { + // eslint-disable-next-line es/no-object-defineproperty -- safe + Object.defineProperty([], 'length', { writable: false }).push(); + } catch (error) { + return error instanceof TypeError; + } +}; + +var FORCED = INCORRECT_TO_LENGTH || !properErrorOnNonWritableLength(); + +// `Array.prototype.push` method +// https://tc39.es/ecma262/#sec-array.prototype.push +$({ target: 'Array', proto: true, arity: 1, forced: FORCED }, { + // eslint-disable-next-line no-unused-vars -- required for `.length` + push: function push(item) { + var O = toObject(this); + var len = lengthOfArrayLike(O); + var argCount = arguments.length; + doesNotExceedSafeInteger(len + argCount); + for (var i = 0; i < argCount; i++) { + O[len] = arguments[i]; + len++; + } + setArrayLength(O, len); + return len; + } +}); + + +/***/ }), + +/***/ 4628: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var newPromiseCapabilityModule = __webpack_require__(6043); + +// `Promise.withResolvers` method +// https://github.com/tc39/proposal-promise-with-resolvers +$({ target: 'Promise', stat: true }, { + withResolvers: function withResolvers() { + var promiseCapability = newPromiseCapabilityModule.f(this); + return { + promise: promiseCapability.promise, + resolve: promiseCapability.resolve, + reject: promiseCapability.reject + }; + } +}); + + +/***/ }), + +/***/ 9479: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var global = __webpack_require__(4475); +var DESCRIPTORS = __webpack_require__(3724); +var defineBuiltInAccessor = __webpack_require__(2106); +var regExpFlags = __webpack_require__(7979); +var fails = __webpack_require__(9039); + +// babel-minify and Closure Compiler transpiles RegExp('.', 'd') -> /./d and it causes SyntaxError +var RegExp = global.RegExp; +var RegExpPrototype = RegExp.prototype; + +var FORCED = DESCRIPTORS && fails(function () { + var INDICES_SUPPORT = true; + try { + RegExp('.', 'd'); + } catch (error) { + INDICES_SUPPORT = false; + } + + var O = {}; + // modern V8 bug + var calls = ''; + var expected = INDICES_SUPPORT ? 'dgimsy' : 'gimsy'; + + var addGetter = function (key, chr) { + // eslint-disable-next-line es/no-object-defineproperty -- safe + Object.defineProperty(O, key, { get: function () { + calls += chr; + return true; + } }); + }; + + var pairs = { + dotAll: 's', + global: 'g', + ignoreCase: 'i', + multiline: 'm', + sticky: 'y' + }; + + if (INDICES_SUPPORT) pairs.hasIndices = 'd'; + + for (var key in pairs) addGetter(key, pairs[key]); + + // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe + var result = Object.getOwnPropertyDescriptor(RegExpPrototype, 'flags').get.call(O); + + return result !== expected || calls !== expected; +}); + +// `RegExp.prototype.flags` getter +// https://tc39.es/ecma262/#sec-get-regexp.prototype.flags +if (FORCED) defineBuiltInAccessor(RegExpPrototype, 'flags', { + configurable: true, + get: regExpFlags +}); + + +/***/ }), + +/***/ 7642: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var difference = __webpack_require__(3440); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.difference` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('difference') }, { + difference: difference +}); + + +/***/ }), + +/***/ 8004: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var fails = __webpack_require__(9039); +var intersection = __webpack_require__(8750); +var setMethodAcceptSetLike = __webpack_require__(4916); + +var INCORRECT = !setMethodAcceptSetLike('intersection') || fails(function () { + // eslint-disable-next-line es/no-array-from, es/no-set -- testing + return String(Array.from(new Set([1, 2, 3]).intersection(new Set([3, 2])))) !== '3,2'; +}); + +// `Set.prototype.intersection` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: INCORRECT }, { + intersection: intersection +}); + + +/***/ }), + +/***/ 3853: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var isDisjointFrom = __webpack_require__(4449); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.isDisjointFrom` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('isDisjointFrom') }, { + isDisjointFrom: isDisjointFrom +}); + + +/***/ }), + +/***/ 5876: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var isSubsetOf = __webpack_require__(3838); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.isSubsetOf` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('isSubsetOf') }, { + isSubsetOf: isSubsetOf +}); + + +/***/ }), + +/***/ 2475: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var isSupersetOf = __webpack_require__(8527); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.isSupersetOf` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('isSupersetOf') }, { + isSupersetOf: isSupersetOf +}); + + +/***/ }), + +/***/ 5024: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var symmetricDifference = __webpack_require__(3650); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.symmetricDifference` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('symmetricDifference') }, { + symmetricDifference: symmetricDifference +}); + + +/***/ }), + +/***/ 1698: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var union = __webpack_require__(4204); +var setMethodAcceptSetLike = __webpack_require__(4916); + +// `Set.prototype.union` method +// https://github.com/tc39/proposal-set-methods +$({ target: 'Set', proto: true, real: true, forced: !setMethodAcceptSetLike('union') }, { + union: union +}); + + +/***/ }), + +/***/ 7467: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var arrayToReversed = __webpack_require__(7628); +var ArrayBufferViewCore = __webpack_require__(4644); + +var aTypedArray = ArrayBufferViewCore.aTypedArray; +var exportTypedArrayMethod = ArrayBufferViewCore.exportTypedArrayMethod; +var getTypedArrayConstructor = ArrayBufferViewCore.getTypedArrayConstructor; + +// `%TypedArray%.prototype.toReversed` method +// https://tc39.es/ecma262/#sec-%typedarray%.prototype.toreversed +exportTypedArrayMethod('toReversed', function toReversed() { + return arrayToReversed(aTypedArray(this), getTypedArrayConstructor(this)); +}); + + +/***/ }), + +/***/ 4732: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var ArrayBufferViewCore = __webpack_require__(4644); +var uncurryThis = __webpack_require__(9504); +var aCallable = __webpack_require__(9306); +var arrayFromConstructorAndList = __webpack_require__(5370); + +var aTypedArray = ArrayBufferViewCore.aTypedArray; +var getTypedArrayConstructor = ArrayBufferViewCore.getTypedArrayConstructor; +var exportTypedArrayMethod = ArrayBufferViewCore.exportTypedArrayMethod; +var sort = uncurryThis(ArrayBufferViewCore.TypedArrayPrototype.sort); + +// `%TypedArray%.prototype.toSorted` method +// https://tc39.es/ecma262/#sec-%typedarray%.prototype.tosorted +exportTypedArrayMethod('toSorted', function toSorted(compareFn) { + if (compareFn !== undefined) aCallable(compareFn); + var O = aTypedArray(this); + var A = arrayFromConstructorAndList(getTypedArrayConstructor(O), O); + return sort(A, compareFn); +}); + + +/***/ }), + +/***/ 9577: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var arrayWith = __webpack_require__(9928); +var ArrayBufferViewCore = __webpack_require__(4644); +var isBigIntArray = __webpack_require__(1108); +var toIntegerOrInfinity = __webpack_require__(1291); +var toBigInt = __webpack_require__(5854); + +var aTypedArray = ArrayBufferViewCore.aTypedArray; +var getTypedArrayConstructor = ArrayBufferViewCore.getTypedArrayConstructor; +var exportTypedArrayMethod = ArrayBufferViewCore.exportTypedArrayMethod; + +var PROPER_ORDER = !!function () { + try { + // eslint-disable-next-line no-throw-literal, es/no-typed-arrays, es/no-array-prototype-with -- required for testing + new Int8Array(1)['with'](2, { valueOf: function () { throw 8; } }); + } catch (error) { + // some early implementations, like WebKit, does not follow the final semantic + // https://github.com/tc39/proposal-change-array-by-copy/pull/86 + return error === 8; + } +}(); + +// `%TypedArray%.prototype.with` method +// https://tc39.es/ecma262/#sec-%typedarray%.prototype.with +exportTypedArrayMethod('with', { 'with': function (index, value) { + var O = aTypedArray(this); + var relativeIndex = toIntegerOrInfinity(index); + var actualValue = isBigIntArray(O) ? toBigInt(value) : +value; + return arrayWith(O, getTypedArrayConstructor(O), relativeIndex, actualValue); +} }['with'], !PROPER_ORDER); + + +/***/ }), + +/***/ 8992: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var global = __webpack_require__(4475); +var anInstance = __webpack_require__(679); +var anObject = __webpack_require__(8551); +var isCallable = __webpack_require__(4901); +var getPrototypeOf = __webpack_require__(2787); +var defineBuiltInAccessor = __webpack_require__(2106); +var createProperty = __webpack_require__(4659); +var fails = __webpack_require__(9039); +var hasOwn = __webpack_require__(9297); +var wellKnownSymbol = __webpack_require__(8227); +var IteratorPrototype = (__webpack_require__(7657).IteratorPrototype); +var DESCRIPTORS = __webpack_require__(3724); +var IS_PURE = __webpack_require__(6395); + +var CONSTRUCTOR = 'constructor'; +var ITERATOR = 'Iterator'; +var TO_STRING_TAG = wellKnownSymbol('toStringTag'); + +var $TypeError = TypeError; +var NativeIterator = global[ITERATOR]; + +// FF56- have non-standard global helper `Iterator` +var FORCED = IS_PURE + || !isCallable(NativeIterator) + || NativeIterator.prototype !== IteratorPrototype + // FF44- non-standard `Iterator` passes previous tests + || !fails(function () { NativeIterator({}); }); + +var IteratorConstructor = function Iterator() { + anInstance(this, IteratorPrototype); + if (getPrototypeOf(this) === IteratorPrototype) throw new $TypeError('Abstract class Iterator not directly constructable'); +}; + +var defineIteratorPrototypeAccessor = function (key, value) { + if (DESCRIPTORS) { + defineBuiltInAccessor(IteratorPrototype, key, { + configurable: true, + get: function () { + return value; + }, + set: function (replacement) { + anObject(this); + if (this === IteratorPrototype) throw new $TypeError("You can't redefine this property"); + if (hasOwn(this, key)) this[key] = replacement; + else createProperty(this, key, replacement); + } + }); + } else IteratorPrototype[key] = value; +}; + +if (!hasOwn(IteratorPrototype, TO_STRING_TAG)) defineIteratorPrototypeAccessor(TO_STRING_TAG, ITERATOR); + +if (FORCED || !hasOwn(IteratorPrototype, CONSTRUCTOR) || IteratorPrototype[CONSTRUCTOR] === Object) { + defineIteratorPrototypeAccessor(CONSTRUCTOR, IteratorConstructor); +} + +IteratorConstructor.prototype = IteratorPrototype; + +// `Iterator` constructor +// https://github.com/tc39/proposal-iterator-helpers +$({ global: true, constructor: true, forced: FORCED }, { + Iterator: IteratorConstructor +}); + + +/***/ }), + +/***/ 3215: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +// `Iterator.prototype.every` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + every: function every(predicate) { + anObject(this); + aCallable(predicate); + var record = getIteratorDirect(this); + var counter = 0; + return !iterate(record, function (value, stop) { + if (!predicate(value, counter++)) return stop(); + }, { IS_RECORD: true, INTERRUPTED: true }).stopped; + } +}); + + +/***/ }), + +/***/ 4520: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var call = __webpack_require__(9565); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); +var createIteratorProxy = __webpack_require__(9462); +var callWithSafeIterationClosing = __webpack_require__(6319); +var IS_PURE = __webpack_require__(6395); + +var IteratorProxy = createIteratorProxy(function () { + var iterator = this.iterator; + var predicate = this.predicate; + var next = this.next; + var result, done, value; + while (true) { + result = anObject(call(next, iterator)); + done = this.done = !!result.done; + if (done) return; + value = result.value; + if (callWithSafeIterationClosing(iterator, predicate, [value, this.counter++], true)) return value; + } +}); + +// `Iterator.prototype.filter` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true, forced: IS_PURE }, { + filter: function filter(predicate) { + anObject(this); + aCallable(predicate); + return new IteratorProxy(getIteratorDirect(this), { + predicate: predicate + }); + } +}); + + +/***/ }), + +/***/ 2577: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +// `Iterator.prototype.find` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + find: function find(predicate) { + anObject(this); + aCallable(predicate); + var record = getIteratorDirect(this); + var counter = 0; + return iterate(record, function (value, stop) { + if (predicate(value, counter++)) return stop(value); + }, { IS_RECORD: true, INTERRUPTED: true }).result; + } +}); + + +/***/ }), + +/***/ 3949: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +// `Iterator.prototype.forEach` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + forEach: function forEach(fn) { + anObject(this); + aCallable(fn); + var record = getIteratorDirect(this); + var counter = 0; + iterate(record, function (value) { + fn(value, counter++); + }, { IS_RECORD: true }); + } +}); + + +/***/ }), + +/***/ 1454: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var map = __webpack_require__(713); +var IS_PURE = __webpack_require__(6395); + +// `Iterator.prototype.map` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true, forced: IS_PURE }, { + map: map +}); + + +/***/ }), + +/***/ 8872: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +var $TypeError = TypeError; + +// `Iterator.prototype.reduce` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + reduce: function reduce(reducer /* , initialValue */) { + anObject(this); + aCallable(reducer); + var record = getIteratorDirect(this); + var noInitial = arguments.length < 2; + var accumulator = noInitial ? undefined : arguments[1]; + var counter = 0; + iterate(record, function (value) { + if (noInitial) { + noInitial = false; + accumulator = value; + } else { + accumulator = reducer(accumulator, value, counter); + } + counter++; + }, { IS_RECORD: true }); + if (noInitial) throw new $TypeError('Reduce of empty iterator with no initial value'); + return accumulator; + } +}); + + +/***/ }), + +/***/ 7550: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var iterate = __webpack_require__(2652); +var aCallable = __webpack_require__(9306); +var anObject = __webpack_require__(8551); +var getIteratorDirect = __webpack_require__(1767); + +// `Iterator.prototype.some` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + some: function some(predicate) { + anObject(this); + aCallable(predicate); + var record = getIteratorDirect(this); + var counter = 0; + return iterate(record, function (value, stop) { + if (predicate(value, counter++)) return stop(); + }, { IS_RECORD: true, INTERRUPTED: true }).stopped; + } +}); + + +/***/ }), + +/***/ 1795: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var anObject = __webpack_require__(8551); +var iterate = __webpack_require__(2652); +var getIteratorDirect = __webpack_require__(1767); + +var push = [].push; + +// `Iterator.prototype.toArray` method +// https://github.com/tc39/proposal-iterator-helpers +$({ target: 'Iterator', proto: true, real: true }, { + toArray: function toArray() { + var result = []; + iterate(getIteratorDirect(anObject(this)), push, { that: result, IS_RECORD: true }); + return result; + } +}); + + +/***/ }), + +/***/ 3375: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(7642); + + +/***/ }), + +/***/ 9225: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(8004); + + +/***/ }), + +/***/ 3972: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(3853); + + +/***/ }), + +/***/ 9209: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(5876); + + +/***/ }), + +/***/ 5714: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(2475); + + +/***/ }), + +/***/ 7561: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(5024); + + +/***/ }), + +/***/ 6197: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +// TODO: Remove from `core-js@4` +__webpack_require__(1698); + + +/***/ }), + +/***/ 4979: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var global = __webpack_require__(4475); +var getBuiltIn = __webpack_require__(7751); +var createPropertyDescriptor = __webpack_require__(6980); +var defineProperty = (__webpack_require__(4913).f); +var hasOwn = __webpack_require__(9297); +var anInstance = __webpack_require__(679); +var inheritIfRequired = __webpack_require__(3167); +var normalizeStringArgument = __webpack_require__(2603); +var DOMExceptionConstants = __webpack_require__(5002); +var clearErrorStack = __webpack_require__(6193); +var DESCRIPTORS = __webpack_require__(3724); +var IS_PURE = __webpack_require__(6395); + +var DOM_EXCEPTION = 'DOMException'; +var Error = getBuiltIn('Error'); +var NativeDOMException = getBuiltIn(DOM_EXCEPTION); + +var $DOMException = function DOMException() { + anInstance(this, DOMExceptionPrototype); + var argumentsLength = arguments.length; + var message = normalizeStringArgument(argumentsLength < 1 ? undefined : arguments[0]); + var name = normalizeStringArgument(argumentsLength < 2 ? undefined : arguments[1], 'Error'); + var that = new NativeDOMException(message, name); + var error = new Error(message); + error.name = DOM_EXCEPTION; + defineProperty(that, 'stack', createPropertyDescriptor(1, clearErrorStack(error.stack, 1))); + inheritIfRequired(that, this, $DOMException); + return that; +}; + +var DOMExceptionPrototype = $DOMException.prototype = NativeDOMException.prototype; + +var ERROR_HAS_STACK = 'stack' in new Error(DOM_EXCEPTION); +var DOM_EXCEPTION_HAS_STACK = 'stack' in new NativeDOMException(1, 2); + +// eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe +var descriptor = NativeDOMException && DESCRIPTORS && Object.getOwnPropertyDescriptor(global, DOM_EXCEPTION); + +// Bun ~ 0.1.1 DOMException have incorrect descriptor and we can't redefine it +// https://github.com/Jarred-Sumner/bun/issues/399 +var BUGGY_DESCRIPTOR = !!descriptor && !(descriptor.writable && descriptor.configurable); + +var FORCED_CONSTRUCTOR = ERROR_HAS_STACK && !BUGGY_DESCRIPTOR && !DOM_EXCEPTION_HAS_STACK; + +// `DOMException` constructor patch for `.stack` where it's required +// https://webidl.spec.whatwg.org/#es-DOMException-specialness +$({ global: true, constructor: true, forced: IS_PURE || FORCED_CONSTRUCTOR }, { // TODO: fix export logic + DOMException: FORCED_CONSTRUCTOR ? $DOMException : NativeDOMException +}); + +var PolyfilledDOMException = getBuiltIn(DOM_EXCEPTION); +var PolyfilledDOMExceptionPrototype = PolyfilledDOMException.prototype; + +if (PolyfilledDOMExceptionPrototype.constructor !== PolyfilledDOMException) { + if (!IS_PURE) { + defineProperty(PolyfilledDOMExceptionPrototype, 'constructor', createPropertyDescriptor(1, PolyfilledDOMException)); + } + + for (var key in DOMExceptionConstants) if (hasOwn(DOMExceptionConstants, key)) { + var constant = DOMExceptionConstants[key]; + var constantName = constant.s; + if (!hasOwn(PolyfilledDOMException, constantName)) { + defineProperty(PolyfilledDOMException, constantName, createPropertyDescriptor(6, constant.c)); + } + } +} + + +/***/ }), + +/***/ 3611: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var $ = __webpack_require__(6518); +var global = __webpack_require__(4475); +var defineBuiltInAccessor = __webpack_require__(2106); +var DESCRIPTORS = __webpack_require__(3724); + +var $TypeError = TypeError; +// eslint-disable-next-line es/no-object-defineproperty -- safe +var defineProperty = Object.defineProperty; +var INCORRECT_VALUE = global.self !== global; + +// `self` getter +// https://html.spec.whatwg.org/multipage/window-object.html#dom-self +try { + if (DESCRIPTORS) { + // eslint-disable-next-line es/no-object-getownpropertydescriptor -- safe + var descriptor = Object.getOwnPropertyDescriptor(global, 'self'); + // some engines have `self`, but with incorrect descriptor + // https://github.com/denoland/deno/issues/15765 + if (INCORRECT_VALUE || !descriptor || !descriptor.get || !descriptor.enumerable) { + defineBuiltInAccessor(global, 'self', { + get: function self() { + return global; + }, + set: function self(value) { + if (this !== global) throw new $TypeError('Illegal invocation'); + defineProperty(global, 'self', { + value: value, + writable: true, + configurable: true, + enumerable: true + }); + }, + configurable: true, + enumerable: true + }); + } + } else $({ global: true, simple: true, forced: INCORRECT_VALUE }, { + self: global + }); +} catch (error) { /* empty */ } + + +/***/ }), + +/***/ 4603: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var defineBuiltIn = __webpack_require__(6840); +var uncurryThis = __webpack_require__(9504); +var toString = __webpack_require__(655); +var validateArgumentsLength = __webpack_require__(2812); + +var $URLSearchParams = URLSearchParams; +var URLSearchParamsPrototype = $URLSearchParams.prototype; +var append = uncurryThis(URLSearchParamsPrototype.append); +var $delete = uncurryThis(URLSearchParamsPrototype['delete']); +var forEach = uncurryThis(URLSearchParamsPrototype.forEach); +var push = uncurryThis([].push); +var params = new $URLSearchParams('a=1&a=2&b=3'); + +params['delete']('a', 1); +// `undefined` case is a Chromium 117 bug +// https://bugs.chromium.org/p/v8/issues/detail?id=14222 +params['delete']('b', undefined); + +if (params + '' !== 'a=2') { + defineBuiltIn(URLSearchParamsPrototype, 'delete', function (name /* , value */) { + var length = arguments.length; + var $value = length < 2 ? undefined : arguments[1]; + if (length && $value === undefined) return $delete(this, name); + var entries = []; + forEach(this, function (v, k) { // also validates `this` + push(entries, { key: k, value: v }); + }); + validateArgumentsLength(length, 1); + var key = toString(name); + var value = toString($value); + var index = 0; + var dindex = 0; + var found = false; + var entriesLength = entries.length; + var entry; + while (index < entriesLength) { + entry = entries[index++]; + if (found || entry.key === key) { + found = true; + $delete(this, entry.key); + } else dindex++; + } + while (dindex < entriesLength) { + entry = entries[dindex++]; + if (!(entry.key === key && entry.value === value)) append(this, entry.key, entry.value); + } + }, { enumerable: true, unsafe: true }); +} + + +/***/ }), + +/***/ 7566: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var defineBuiltIn = __webpack_require__(6840); +var uncurryThis = __webpack_require__(9504); +var toString = __webpack_require__(655); +var validateArgumentsLength = __webpack_require__(2812); + +var $URLSearchParams = URLSearchParams; +var URLSearchParamsPrototype = $URLSearchParams.prototype; +var getAll = uncurryThis(URLSearchParamsPrototype.getAll); +var $has = uncurryThis(URLSearchParamsPrototype.has); +var params = new $URLSearchParams('a=1'); + +// `undefined` case is a Chromium 117 bug +// https://bugs.chromium.org/p/v8/issues/detail?id=14222 +if (params.has('a', 2) || !params.has('a', undefined)) { + defineBuiltIn(URLSearchParamsPrototype, 'has', function has(name /* , value */) { + var length = arguments.length; + var $value = length < 2 ? undefined : arguments[1]; + if (length && $value === undefined) return $has(this, name); + var values = getAll(this, name); // also validates `this` + validateArgumentsLength(length, 1); + var value = toString($value); + var index = 0; + while (index < values.length) { + if (values[index++] === value) return true; + } return false; + }, { enumerable: true, unsafe: true }); +} + + +/***/ }), + +/***/ 8721: +/***/ ((__unused_webpack_module, __unused_webpack_exports, __webpack_require__) => { + + +var DESCRIPTORS = __webpack_require__(3724); +var uncurryThis = __webpack_require__(9504); +var defineBuiltInAccessor = __webpack_require__(2106); + +var URLSearchParamsPrototype = URLSearchParams.prototype; +var forEach = uncurryThis(URLSearchParamsPrototype.forEach); + +// `URLSearchParams.prototype.size` getter +// https://github.com/whatwg/url/pull/734 +if (DESCRIPTORS && !('size' in URLSearchParamsPrototype)) { + defineBuiltInAccessor(URLSearchParamsPrototype, 'size', { + get: function size() { + var count = 0; + forEach(this, function () { count++; }); + return count; + }, + configurable: true, + enumerable: true + }); +} + + +/***/ }) + +/******/ }); +/************************************************************************/ +/******/ // The module cache +/******/ var __webpack_module_cache__ = {}; +/******/ +/******/ // The require function +/******/ function __webpack_require__(moduleId) { +/******/ // Check if module is in cache +/******/ var cachedModule = __webpack_module_cache__[moduleId]; +/******/ if (cachedModule !== undefined) { +/******/ return cachedModule.exports; +/******/ } +/******/ // Create a new module (and put it into the cache) +/******/ var module = __webpack_module_cache__[moduleId] = { +/******/ // no module.id needed +/******/ // no module.loaded needed +/******/ exports: {} +/******/ }; +/******/ +/******/ // Execute the module function +/******/ __webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__); +/******/ +/******/ // Return the exports of the module +/******/ return module.exports; +/******/ } /******/ /************************************************************************/ /******/ /* webpack/runtime/define property getters */ @@ -49,7 +4540,63 @@ __webpack_require__.d(__webpack_exports__, { WorkerMessageHandler: () => (/* reexport */ WorkerMessageHandler) }); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array.push.js +var es_array_push = __webpack_require__(4114); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array-buffer.detached.js +var es_array_buffer_detached = __webpack_require__(6573); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array-buffer.transfer.js +var es_array_buffer_transfer = __webpack_require__(8100); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.array-buffer.transfer-to-fixed-length.js +var es_array_buffer_transfer_to_fixed_length = __webpack_require__(7936); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.promise.with-resolvers.js +var es_promise_with_resolvers = __webpack_require__(4628); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.typed-array.to-reversed.js +var es_typed_array_to_reversed = __webpack_require__(7467); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.typed-array.to-sorted.js +var es_typed_array_to_sorted = __webpack_require__(4732); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.typed-array.with.js +var es_typed_array_with = __webpack_require__(9577); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.constructor.js +var esnext_iterator_constructor = __webpack_require__(8992); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.for-each.js +var esnext_iterator_for_each = __webpack_require__(3949); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.map.js +var esnext_iterator_map = __webpack_require__(1454); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.some.js +var esnext_iterator_some = __webpack_require__(7550); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.difference.v2.js +var esnext_set_difference_v2 = __webpack_require__(3375); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.intersection.v2.js +var esnext_set_intersection_v2 = __webpack_require__(9225); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.is-disjoint-from.v2.js +var esnext_set_is_disjoint_from_v2 = __webpack_require__(3972); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.is-subset-of.v2.js +var esnext_set_is_subset_of_v2 = __webpack_require__(9209); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.is-superset-of.v2.js +var esnext_set_is_superset_of_v2 = __webpack_require__(5714); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.symmetric-difference.v2.js +var esnext_set_symmetric_difference_v2 = __webpack_require__(7561); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.set.union.v2.js +var esnext_set_union_v2 = __webpack_require__(6197); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.self.js +var web_self = __webpack_require__(3611); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.url-search-params.delete.js +var web_url_search_params_delete = __webpack_require__(4603); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.url-search-params.has.js +var web_url_search_params_has = __webpack_require__(7566); +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.url-search-params.size.js +var web_url_search_params_size = __webpack_require__(8721); ;// CONCATENATED MODULE: ./src/shared/util.js + + + + + + + + + + const isNodeJS = typeof process === "object" && process + "" === "[object process]" && !process.versions.nw && !(process.versions.electron && process.type && process.type !== "browser"); const IDENTITY_MATRIX = [1, 0, 0, 1, 0, 0]; const FONT_IDENTITY_MATRIX = [0.001, 0, 0, 0.001, 0, 0]; @@ -65,6 +4612,7 @@ const RenderingIntentFlag = { ANNOTATIONS_FORMS: 0x10, ANNOTATIONS_STORAGE: 0x20, ANNOTATIONS_DISABLE: 0x40, + IS_EDITING: 0x80, OPLIST: 0x100 }; const AnnotationMode = { @@ -320,7 +4868,9 @@ const OPS = { paintImageXObjectRepeat: 88, paintImageMaskXObjectRepeat: 89, paintSolidColorImageMask: 90, - constructPath: 91 + constructPath: 91, + setStrokeTransparent: 92, + setFillTransparent: 93 }; const PasswordResponses = { NEED_PASSWORD: 1, @@ -785,8 +5335,18 @@ const FontRenderOps = { TRANSLATE: 8 }; +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.every.js +var esnext_iterator_every = __webpack_require__(3215); ;// CONCATENATED MODULE: ./src/core/primitives.js + + + + + + + + const CIRCULAR_REF = Symbol("CIRCULAR_REF"); const EOF = Symbol("EOF"); let CmdCache = Object.create(null); @@ -953,6 +5513,9 @@ class Dict { } return dict; } + delete(key) { + delete this._map[key]; + } } class Ref { constructor(num, gen) { @@ -1134,6 +5697,23 @@ class BaseStream { + + + + + + + + + + + + + + + + + const PDF_VERSION_REGEXP = /^[1-9]\.\d$/; function getLookupTableFactory(initializer) { let lookup; @@ -1524,6 +6104,9 @@ function getNewAnnotationsMap(annotationStorage) { } return newAnnotationsByPage.size > 0 ? newAnnotationsByPage : null; } +function stringToAsciiOrUTF16BE(str) { + return isAscii(str) ? str : stringToUTF16String(str, true); +} function isAscii(str) { return /^[\x00-\x7F]*$/.test(str); } @@ -1565,6 +6148,12 @@ function getSizeInBytes(x) { ;// CONCATENATED MODULE: ./src/core/stream.js + + + + + + class Stream extends BaseStream { constructor(arrayBuffer, start, length, dict) { super(); @@ -1634,6 +6223,21 @@ class NullStream extends Stream { + + + + + + + + + + + + + + + class ChunkedStream extends Stream { constructor(length, chunkSize, manager) { super(new Uint8Array(length), 0, length, null); @@ -2057,11 +6661,25 @@ class ChunkedStreamManager { } } +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.filter.js +var esnext_iterator_filter = __webpack_require__(4520); +// EXTERNAL MODULE: ./node_modules/core-js/modules/es.regexp.flags.js +var es_regexp_flags = __webpack_require__(9479); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.find.js +var esnext_iterator_find = __webpack_require__(2577); +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.reduce.js +var esnext_iterator_reduce = __webpack_require__(8872); ;// CONCATENATED MODULE: ./src/core/colorspace.js + + + + + + function resizeRgbImage(src, dest, w1, h1, w2, h2, alpha01) { const COMPONENTS = 3; alpha01 = alpha01 !== 1 ? 0 : alpha01; @@ -2271,7 +6889,8 @@ class ColorSpace { } } } - throw new FormatError(`Unrecognized ColorSpace: ${cs.name}`); + warn(`Unrecognized ColorSpace: ${cs.name}`); + return this.singletons.gray; } } if (Array.isArray(cs)) { @@ -2346,10 +6965,12 @@ class ColorSpace { const range = params.getArray("Range"); return new LabCS(whitePoint, blackPoint, range); default: - throw new FormatError(`Unimplemented ColorSpace object: ${mode}`); + warn(`Unimplemented ColorSpace object: ${mode}`); + return this.singletons.gray; } } - throw new FormatError(`Unrecognized ColorSpace object: ${cs}`); + warn(`Unrecognized ColorSpace object: ${cs}`); + return this.singletons.gray; } static isDefaultDecode(decode, numComps) { if (!Array.isArray(decode)) { @@ -2880,6 +7501,12 @@ class LabCS extends ColorSpace { ;// CONCATENATED MODULE: ./src/core/binary_cmap.js + + + + + + function hexToInt(a, size) { let n = 0; for (let i = 0; i <= size; i++) { @@ -3138,6 +7765,13 @@ class BinaryCMapReader { ;// CONCATENATED MODULE: ./src/core/decode_stream.js + + + + + + + const emptyBuffer = new Uint8Array(0); class DecodeStream extends BaseStream { constructor(maybeMinBufferLength) { @@ -3280,6 +7914,12 @@ class StreamsSequenceStream extends DecodeStream { ;// CONCATENATED MODULE: ./src/core/ascii_85_stream.js + + + + + + class Ascii85Stream extends DecodeStream { constructor(str, maybeLength) { if (maybeLength) { @@ -3397,6 +8037,12 @@ class AsciiHexStream extends DecodeStream { ;// CONCATENATED MODULE: ./src/core/ccitt.js + + + + + + const ccittEOL = -2; const ccittEOF = -1; const twoDimPass = 0; @@ -3949,6 +8595,13 @@ class CCITTFaxStream extends DecodeStream { + + + + + + + const codeLenCodeMap = new Int32Array([16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15]); const lengthDecode = new Int32Array([0x00003, 0x00004, 0x00005, 0x00006, 0x00007, 0x00008, 0x00009, 0x0000a, 0x1000b, 0x1000d, 0x1000f, 0x10011, 0x20013, 0x20017, 0x2001b, 0x2001f, 0x30023, 0x3002b, 0x30033, 0x3003b, 0x40043, 0x40053, 0x40063, 0x40073, 0x50083, 0x500a3, 0x500c3, 0x500e3, 0x00102, 0x00102, 0x00102]); const distDecode = new Int32Array([0x00001, 0x00002, 0x00003, 0x00004, 0x10005, 0x10007, 0x20009, 0x2000d, 0x30011, 0x30019, 0x40021, 0x40031, 0x50041, 0x50061, 0x60081, 0x600c1, 0x70101, 0x70181, 0x80201, 0x80301, 0x90401, 0x90601, 0xa0801, 0xa0c01, 0xb1001, 0xb1801, 0xc2001, 0xc3001, 0xd4001, 0xd6001]); @@ -4565,6 +9218,13 @@ class ArithmeticDecoder { + + + + + + + class Jbig2Error extends BaseException { constructor(msg) { super(msg, "Jbig2Error"); @@ -6317,6 +10977,7 @@ class Jbig2Image { + class Jbig2Stream extends DecodeStream { constructor(stream, maybeLength, params) { super(maybeLength); @@ -6372,6 +11033,12 @@ class Jbig2Stream extends DecodeStream { ;// CONCATENATED MODULE: ./src/shared/image_utils.js + + + + + + function convertToRGBA(params) { switch (params.kind) { case ImageKind.GRAYSCALE_1BPP: @@ -6481,6 +11148,13 @@ function grayToRGBA(src, dest) { + + + + + + + class JpegError extends BaseException { constructor(msg) { super(msg, "JpegError"); @@ -7554,6 +12228,12 @@ class JpegImage { + + + + + + class JpegStream extends DecodeStream { constructor(stream, maybeLength, params) { super(maybeLength); @@ -7629,7 +12309,20 @@ class JpegStream extends DecodeStream { } } +// EXTERNAL MODULE: ./node_modules/core-js/modules/web.dom-exception.stack.js +var web_dom_exception_stack = __webpack_require__(4979); ;// CONCATENATED MODULE: ./external/openjpeg/openjpeg.js + + + + + + + + + + + var OpenJPEG = (() => { var _scriptName = typeof document != 'undefined' ? document.currentScript?.src : undefined; return function (moduleArg = {}) { @@ -8173,6 +12866,9 @@ var OpenJPEG = (() => { + + + class JpxError extends BaseException { constructor(msg) { super(msg, "JpxError"); @@ -8256,6 +12952,12 @@ class JpxStream extends DecodeStream { ;// CONCATENATED MODULE: ./src/core/lzw_stream.js + + + + + + class LZWStream extends DecodeStream { constructor(str, maybeLength, earlyChange) { super(maybeLength); @@ -8375,6 +13077,12 @@ class LZWStream extends DecodeStream { + + + + + + class PredictorStream extends DecodeStream { constructor(str, maybeLength, params) { super(maybeLength); @@ -8607,6 +13315,15 @@ class RunLengthStream extends DecodeStream { + + + + + + + + + @@ -9676,6 +14393,9 @@ class Linearization { + + + const BUILT_IN_CMAPS = ["Adobe-GB1-UCS2", "Adobe-CNS1-UCS2", "Adobe-Japan1-UCS2", "Adobe-Korea1-UCS2", "78-EUC-H", "78-EUC-V", "78-H", "78-RKSJ-H", "78-RKSJ-V", "78-V", "78ms-RKSJ-H", "78ms-RKSJ-V", "83pv-RKSJ-H", "90ms-RKSJ-H", "90ms-RKSJ-V", "90msp-RKSJ-H", "90msp-RKSJ-V", "90pv-RKSJ-H", "90pv-RKSJ-V", "Add-H", "Add-RKSJ-H", "Add-RKSJ-V", "Add-V", "Adobe-CNS1-0", "Adobe-CNS1-1", "Adobe-CNS1-2", "Adobe-CNS1-3", "Adobe-CNS1-4", "Adobe-CNS1-5", "Adobe-CNS1-6", "Adobe-GB1-0", "Adobe-GB1-1", "Adobe-GB1-2", "Adobe-GB1-3", "Adobe-GB1-4", "Adobe-GB1-5", "Adobe-Japan1-0", "Adobe-Japan1-1", "Adobe-Japan1-2", "Adobe-Japan1-3", "Adobe-Japan1-4", "Adobe-Japan1-5", "Adobe-Japan1-6", "Adobe-Korea1-0", "Adobe-Korea1-1", "Adobe-Korea1-2", "B5-H", "B5-V", "B5pc-H", "B5pc-V", "CNS-EUC-H", "CNS-EUC-V", "CNS1-H", "CNS1-V", "CNS2-H", "CNS2-V", "ETHK-B5-H", "ETHK-B5-V", "ETen-B5-H", "ETen-B5-V", "ETenms-B5-H", "ETenms-B5-V", "EUC-H", "EUC-V", "Ext-H", "Ext-RKSJ-H", "Ext-RKSJ-V", "Ext-V", "GB-EUC-H", "GB-EUC-V", "GB-H", "GB-V", "GBK-EUC-H", "GBK-EUC-V", "GBK2K-H", "GBK2K-V", "GBKp-EUC-H", "GBKp-EUC-V", "GBT-EUC-H", "GBT-EUC-V", "GBT-H", "GBT-V", "GBTpc-EUC-H", "GBTpc-EUC-V", "GBpc-EUC-H", "GBpc-EUC-V", "H", "HKdla-B5-H", "HKdla-B5-V", "HKdlb-B5-H", "HKdlb-B5-V", "HKgccs-B5-H", "HKgccs-B5-V", "HKm314-B5-H", "HKm314-B5-V", "HKm471-B5-H", "HKm471-B5-V", "HKscs-B5-H", "HKscs-B5-V", "Hankaku", "Hiragana", "KSC-EUC-H", "KSC-EUC-V", "KSC-H", "KSC-Johab-H", "KSC-Johab-V", "KSC-V", "KSCms-UHC-H", "KSCms-UHC-HW-H", "KSCms-UHC-HW-V", "KSCms-UHC-V", "KSCpc-EUC-H", "KSCpc-EUC-V", "Katakana", "NWP-H", "NWP-V", "RKSJ-H", "RKSJ-V", "Roman", "UniCNS-UCS2-H", "UniCNS-UCS2-V", "UniCNS-UTF16-H", "UniCNS-UTF16-V", "UniCNS-UTF32-H", "UniCNS-UTF32-V", "UniCNS-UTF8-H", "UniCNS-UTF8-V", "UniGB-UCS2-H", "UniGB-UCS2-V", "UniGB-UTF16-H", "UniGB-UTF16-V", "UniGB-UTF32-H", "UniGB-UTF32-V", "UniGB-UTF8-H", "UniGB-UTF8-V", "UniJIS-UCS2-H", "UniJIS-UCS2-HW-H", "UniJIS-UCS2-HW-V", "UniJIS-UCS2-V", "UniJIS-UTF16-H", "UniJIS-UTF16-V", "UniJIS-UTF32-H", "UniJIS-UTF32-V", "UniJIS-UTF8-H", "UniJIS-UTF8-V", "UniJIS2004-UTF16-H", "UniJIS2004-UTF16-V", "UniJIS2004-UTF32-H", "UniJIS2004-UTF32-V", "UniJIS2004-UTF8-H", "UniJIS2004-UTF8-V", "UniJISPro-UCS2-HW-V", "UniJISPro-UCS2-V", "UniJISPro-UTF8-V", "UniJISX0213-UTF32-H", "UniJISX0213-UTF32-V", "UniJISX02132004-UTF32-H", "UniJISX02132004-UTF32-V", "UniKS-UCS2-H", "UniKS-UCS2-V", "UniKS-UTF16-H", "UniKS-UTF16-V", "UniKS-UTF32-H", "UniKS-UTF32-V", "UniKS-UTF8-H", "UniKS-UTF8-V", "V", "WP-Symbol"]; const MAX_MAP_RANGE = 2 ** 24 - 1; class CMap { @@ -10116,6 +14836,8 @@ class CMapFactory { } } +// EXTERNAL MODULE: ./node_modules/core-js/modules/esnext.iterator.to-array.js +var esnext_iterator_to_array = __webpack_require__(1795); ;// CONCATENATED MODULE: ./src/core/charsets.js const ISOAdobeCharset = [".notdef", "space", "exclam", "quotedbl", "numbersign", "dollar", "percent", "ampersand", "quoteright", "parenleft", "parenright", "asterisk", "plus", "comma", "hyphen", "period", "slash", "zero", "one", "two", "three", "four", "five", "six", "seven", "eight", "nine", "colon", "semicolon", "less", "equal", "greater", "question", "at", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z", "bracketleft", "backslash", "bracketright", "asciicircum", "underscore", "quoteleft", "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z", "braceleft", "bar", "braceright", "asciitilde", "exclamdown", "cent", "sterling", "fraction", "yen", "florin", "section", "currency", "quotesingle", "quotedblleft", "guillemotleft", "guilsinglleft", "guilsinglright", "fi", "fl", "endash", "dagger", "daggerdbl", "periodcentered", "paragraph", "bullet", "quotesinglbase", "quotedblbase", "quotedblright", "guillemotright", "ellipsis", "perthousand", "questiondown", "grave", "acute", "circumflex", "tilde", "macron", "breve", "dotaccent", "dieresis", "ring", "cedilla", "hungarumlaut", "ogonek", "caron", "emdash", "AE", "ordfeminine", "Lslash", "Oslash", "OE", "ordmasculine", "ae", "dotlessi", "lslash", "oslash", "oe", "germandbls", "onesuperior", "logicalnot", "mu", "trademark", "Eth", "onehalf", "plusminus", "Thorn", "onequarter", "divide", "brokenbar", "degree", "thorn", "threequarters", "twosuperior", "registered", "minus", "eth", "multiply", "threesuperior", "copyright", "Aacute", "Acircumflex", "Adieresis", "Agrave", "Aring", "Atilde", "Ccedilla", "Eacute", "Ecircumflex", "Edieresis", "Egrave", "Iacute", "Icircumflex", "Idieresis", "Igrave", "Ntilde", "Oacute", "Ocircumflex", "Odieresis", "Ograve", "Otilde", "Scaron", "Uacute", "Ucircumflex", "Udieresis", "Ugrave", "Yacute", "Ydieresis", "Zcaron", "aacute", "acircumflex", "adieresis", "agrave", "aring", "atilde", "ccedilla", "eacute", "ecircumflex", "edieresis", "egrave", "iacute", "icircumflex", "idieresis", "igrave", "ntilde", "oacute", "ocircumflex", "odieresis", "ograve", "otilde", "scaron", "uacute", "ucircumflex", "udieresis", "ugrave", "yacute", "ydieresis", "zcaron"]; const ExpertCharset = [".notdef", "space", "exclamsmall", "Hungarumlautsmall", "dollaroldstyle", "dollarsuperior", "ampersandsmall", "Acutesmall", "parenleftsuperior", "parenrightsuperior", "twodotenleader", "onedotenleader", "comma", "hyphen", "period", "fraction", "zerooldstyle", "oneoldstyle", "twooldstyle", "threeoldstyle", "fouroldstyle", "fiveoldstyle", "sixoldstyle", "sevenoldstyle", "eightoldstyle", "nineoldstyle", "colon", "semicolon", "commasuperior", "threequartersemdash", "periodsuperior", "questionsmall", "asuperior", "bsuperior", "centsuperior", "dsuperior", "esuperior", "isuperior", "lsuperior", "msuperior", "nsuperior", "osuperior", "rsuperior", "ssuperior", "tsuperior", "ff", "fi", "fl", "ffi", "ffl", "parenleftinferior", "parenrightinferior", "Circumflexsmall", "hyphensuperior", "Gravesmall", "Asmall", "Bsmall", "Csmall", "Dsmall", "Esmall", "Fsmall", "Gsmall", "Hsmall", "Ismall", "Jsmall", "Ksmall", "Lsmall", "Msmall", "Nsmall", "Osmall", "Psmall", "Qsmall", "Rsmall", "Ssmall", "Tsmall", "Usmall", "Vsmall", "Wsmall", "Xsmall", "Ysmall", "Zsmall", "colonmonetary", "onefitted", "rupiah", "Tildesmall", "exclamdownsmall", "centoldstyle", "Lslashsmall", "Scaronsmall", "Zcaronsmall", "Dieresissmall", "Brevesmall", "Caronsmall", "Dotaccentsmall", "Macronsmall", "figuredash", "hypheninferior", "Ogoneksmall", "Ringsmall", "Cedillasmall", "onequarter", "onehalf", "threequarters", "questiondownsmall", "oneeighth", "threeeighths", "fiveeighths", "seveneighths", "onethird", "twothirds", "zerosuperior", "onesuperior", "twosuperior", "threesuperior", "foursuperior", "fivesuperior", "sixsuperior", "sevensuperior", "eightsuperior", "ninesuperior", "zeroinferior", "oneinferior", "twoinferior", "threeinferior", "fourinferior", "fiveinferior", "sixinferior", "seveninferior", "eightinferior", "nineinferior", "centinferior", "dollarinferior", "periodinferior", "commainferior", "Agravesmall", "Aacutesmall", "Acircumflexsmall", "Atildesmall", "Adieresissmall", "Aringsmall", "AEsmall", "Ccedillasmall", "Egravesmall", "Eacutesmall", "Ecircumflexsmall", "Edieresissmall", "Igravesmall", "Iacutesmall", "Icircumflexsmall", "Idieresissmall", "Ethsmall", "Ntildesmall", "Ogravesmall", "Oacutesmall", "Ocircumflexsmall", "Otildesmall", "Odieresissmall", "OEsmall", "Oslashsmall", "Ugravesmall", "Uacutesmall", "Ucircumflexsmall", "Udieresissmall", "Yacutesmall", "Thornsmall", "Ydieresissmall"]; @@ -10154,6 +14876,13 @@ function getEncoding(encodingName) { + + + + + + + const MAX_SUBR_NESTING = 10; const CFFStandardStrings = [".notdef", "space", "exclam", "quotedbl", "numbersign", "dollar", "percent", "ampersand", "quoteright", "parenleft", "parenright", "asterisk", "plus", "comma", "hyphen", "period", "slash", "zero", "one", "two", "three", "four", "five", "six", "seven", "eight", "nine", "colon", "semicolon", "less", "equal", "greater", "question", "at", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z", "bracketleft", "backslash", "bracketright", "asciicircum", "underscore", "quoteleft", "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z", "braceleft", "bar", "braceright", "asciitilde", "exclamdown", "cent", "sterling", "fraction", "yen", "florin", "section", "currency", "quotesingle", "quotedblleft", "guillemotleft", "guilsinglleft", "guilsinglright", "fi", "fl", "endash", "dagger", "daggerdbl", "periodcentered", "paragraph", "bullet", "quotesinglbase", "quotedblbase", "quotedblright", "guillemotright", "ellipsis", "perthousand", "questiondown", "grave", "acute", "circumflex", "tilde", "macron", "breve", "dotaccent", "dieresis", "ring", "cedilla", "hungarumlaut", "ogonek", "caron", "emdash", "AE", "ordfeminine", "Lslash", "Oslash", "OE", "ordmasculine", "ae", "dotlessi", "lslash", "oslash", "oe", "germandbls", "onesuperior", "logicalnot", "mu", "trademark", "Eth", "onehalf", "plusminus", "Thorn", "onequarter", "divide", "brokenbar", "degree", "thorn", "threequarters", "twosuperior", "registered", "minus", "eth", "multiply", "threesuperior", "copyright", "Aacute", "Acircumflex", "Adieresis", "Agrave", "Aring", "Atilde", "Ccedilla", "Eacute", "Ecircumflex", "Edieresis", "Egrave", "Iacute", "Icircumflex", "Idieresis", "Igrave", "Ntilde", "Oacute", "Ocircumflex", "Odieresis", "Ograve", "Otilde", "Scaron", "Uacute", "Ucircumflex", "Udieresis", "Ugrave", "Yacute", "Ydieresis", "Zcaron", "aacute", "acircumflex", "adieresis", "agrave", "aring", "atilde", "ccedilla", "eacute", "ecircumflex", "edieresis", "egrave", "iacute", "icircumflex", "idieresis", "igrave", "ntilde", "oacute", "ocircumflex", "odieresis", "ograve", "otilde", "scaron", "uacute", "ucircumflex", "udieresis", "ugrave", "yacute", "ydieresis", "zcaron", "exclamsmall", "Hungarumlautsmall", "dollaroldstyle", "dollarsuperior", "ampersandsmall", "Acutesmall", "parenleftsuperior", "parenrightsuperior", "twodotenleader", "onedotenleader", "zerooldstyle", "oneoldstyle", "twooldstyle", "threeoldstyle", "fouroldstyle", "fiveoldstyle", "sixoldstyle", "sevenoldstyle", "eightoldstyle", "nineoldstyle", "commasuperior", "threequartersemdash", "periodsuperior", "questionsmall", "asuperior", "bsuperior", "centsuperior", "dsuperior", "esuperior", "isuperior", "lsuperior", "msuperior", "nsuperior", "osuperior", "rsuperior", "ssuperior", "tsuperior", "ff", "ffi", "ffl", "parenleftinferior", "parenrightinferior", "Circumflexsmall", "hyphensuperior", "Gravesmall", "Asmall", "Bsmall", "Csmall", "Dsmall", "Esmall", "Fsmall", "Gsmall", "Hsmall", "Ismall", "Jsmall", "Ksmall", "Lsmall", "Msmall", "Nsmall", "Osmall", "Psmall", "Qsmall", "Rsmall", "Ssmall", "Tsmall", "Usmall", "Vsmall", "Wsmall", "Xsmall", "Ysmall", "Zsmall", "colonmonetary", "onefitted", "rupiah", "Tildesmall", "exclamdownsmall", "centoldstyle", "Lslashsmall", "Scaronsmall", "Zcaronsmall", "Dieresissmall", "Brevesmall", "Caronsmall", "Dotaccentsmall", "Macronsmall", "figuredash", "hypheninferior", "Ogoneksmall", "Ringsmall", "Cedillasmall", "questiondownsmall", "oneeighth", "threeeighths", "fiveeighths", "seveneighths", "onethird", "twothirds", "zerosuperior", "foursuperior", "fivesuperior", "sixsuperior", "sevensuperior", "eightsuperior", "ninesuperior", "zeroinferior", "oneinferior", "twoinferior", "threeinferior", "fourinferior", "fiveinferior", "sixinferior", "seveninferior", "eightinferior", "nineinferior", "centinferior", "dollarinferior", "periodinferior", "commainferior", "Agravesmall", "Aacutesmall", "Acircumflexsmall", "Atildesmall", "Adieresissmall", "Aringsmall", "AEsmall", "Ccedillasmall", "Egravesmall", "Eacutesmall", "Ecircumflexsmall", "Edieresissmall", "Igravesmall", "Iacutesmall", "Icircumflexsmall", "Idieresissmall", "Ethsmall", "Ntildesmall", "Ogravesmall", "Oacutesmall", "Ocircumflexsmall", "Otildesmall", "Odieresissmall", "OEsmall", "Oslashsmall", "Ugravesmall", "Uacutesmall", "Ucircumflexsmall", "Udieresissmall", "Yacutesmall", "Thornsmall", "Ydieresissmall", "001.000", "001.001", "001.002", "001.003", "Black", "Bold", "Book", "Light", "Medium", "Regular", "Roman", "Semibold"]; const NUM_STANDARD_CFF_STRINGS = 391; @@ -16267,6 +20996,8 @@ function clearUnicodeCaches() { + + const SEAC_ANALYSIS_ENABLED = true; const FontFlags = { FixedPitch: 1, @@ -16345,6 +21076,41 @@ function type1FontGlyphMapping(properties, builtInEncoding, glyphNames) { function normalizeFontName(name) { return name.replaceAll(/[,_]/g, "-").replaceAll(/\s/g, ""); } +const getVerticalPresentationForm = getLookupTableFactory(t => { + t[0x2013] = 0xfe32; + t[0x2014] = 0xfe31; + t[0x2025] = 0xfe30; + t[0x2026] = 0xfe19; + t[0x3001] = 0xfe11; + t[0x3002] = 0xfe12; + t[0x3008] = 0xfe3f; + t[0x3009] = 0xfe40; + t[0x300a] = 0xfe3d; + t[0x300b] = 0xfe3e; + t[0x300c] = 0xfe41; + t[0x300d] = 0xfe42; + t[0x300e] = 0xfe43; + t[0x300f] = 0xfe44; + t[0x3010] = 0xfe3b; + t[0x3011] = 0xfe3c; + t[0x3014] = 0xfe39; + t[0x3015] = 0xfe3a; + t[0x3016] = 0xfe17; + t[0x3017] = 0xfe18; + t[0xfe4f] = 0xfe34; + t[0xff01] = 0xfe15; + t[0xff08] = 0xfe35; + t[0xff09] = 0xfe36; + t[0xff0c] = 0xfe10; + t[0xff1a] = 0xfe13; + t[0xff1b] = 0xfe14; + t[0xff1f] = 0xfe16; + t[0xff3b] = 0xfe47; + t[0xff3d] = 0xfe48; + t[0xff3f] = 0xfe33; + t[0xff5b] = 0xfe37; + t[0xff5d] = 0xfe38; +}); ;// CONCATENATED MODULE: ./src/core/standard_fonts.js @@ -17374,6 +22140,15 @@ class CFFFont { + + + + + + + + + function getUint32(data, offset) { return (data[offset] << 24 | data[offset + 1] << 16 | data[offset + 2] << 8 | data[offset + 3]) >>> 0; } @@ -21179,6 +25954,16 @@ const getFontBasicMetrics = getLookupTableFactory(function (t) { }); ;// CONCATENATED MODULE: ./src/core/glyf.js + + + + + + + + + + const ON_CURVE_POINT = 1 << 0; const X_SHORT_VECTOR = 1 << 1; const Y_SHORT_VECTOR = 1 << 2; @@ -21712,6 +26497,13 @@ class CompositeGlyph { ;// CONCATENATED MODULE: ./src/core/opentype_file_builder.js + + + + + + + function writeInt16(dest, offset, num) { dest[offset] = num >> 8 & 0xff; dest[offset + 1] = num & 0xff; @@ -21820,6 +26612,13 @@ class OpenTypeFileBuilder { + + + + + + + const HINTING_ENABLED = false; const COMMAND_MAP = { hstem: [1], @@ -22387,6 +27186,7 @@ class Type1Parser { + function findBlock(streamBytes, signature, startIndex) { const streamBytesLength = streamBytes.length; const signatureLength = signature.length; @@ -22672,6 +27472,25 @@ class Type1Font { + + + + + + + + + + + + + + + + + + + const PRIVATE_USE_AREAS = [[0xe000, 0xf8ff], [0x100000, 0x10fffd]]; @@ -24902,6 +29721,36 @@ class Font { builder.addTable("post", createPostTable(properties)); return builder.toArray(); } + get _spaceWidth() { + const possibleSpaceReplacements = ["space", "minus", "one", "i", "I"]; + let width; + for (const glyphName of possibleSpaceReplacements) { + if (glyphName in this.widths) { + width = this.widths[glyphName]; + break; + } + const glyphsUnicodeMap = getGlyphsUnicode(); + const glyphUnicode = glyphsUnicodeMap[glyphName]; + let charcode = 0; + if (this.composite && this.cMap.contains(glyphUnicode)) { + charcode = this.cMap.lookup(glyphUnicode); + if (typeof charcode === "string") { + charcode = convertCidString(glyphUnicode, charcode); + } + } + if (!charcode && this.toUnicode) { + charcode = this.toUnicode.charCodeOf(glyphUnicode); + } + if (charcode <= 0) { + charcode = glyphUnicode; + } + width = this.widths[charcode]; + if (width) { + break; + } + } + return shadow(this, "_spaceWidth", width || this.defaultWidth); + } _charToGlyph(charcode, isSpace = false) { let glyph = this._glyphCache[charcode]; if (glyph?.isSpace === isSpace) { @@ -24930,6 +29779,10 @@ class Font { const glyphName = this.differences[charcode] || this.defaultEncoding[charcode]; if ((glyphName === ".notdef" || glyphName === "") && this.type === "Type1") { fontCharCode = 0x20; + if (glyphName === "") { + width ||= this._spaceWidth; + unicode = String.fromCharCode(fontCharCode); + } } fontCharCode = mapSpecialUnicodeValues(fontCharCode); } @@ -24954,6 +29807,12 @@ class Font { warn(`charToGlyph - invalid fontCharCode: ${fontCharCode}`); } } + if (this.missingFile && this.vertical && fontChar.length === 1) { + const vertical = getVerticalPresentationForm()[fontChar.charCodeAt(0)]; + if (vertical) { + fontChar = unicode = String.fromCharCode(vertical); + } + } glyph = new fonts_Glyph(charcode, fontChar, unicode, accent, width, vmetric, operatorListId, isSpace, isInFont); return this._glyphCache[charcode] = glyph; } @@ -25065,6 +29924,13 @@ class ErrorFont { + + + + + + + const ShadingType = { FUNCTION_BASED: 1, AXIAL: 2, @@ -25985,6 +30851,8 @@ const SegoeuiRegularMetrics = { + + const getXFAFontMap = getLookupTableFactory(function (t) { t["MyriadPro-Regular"] = t["PdfJS-Fallback-Regular"] = { name: "LiberationSans-Regular", @@ -26178,6 +31046,7 @@ function getXfaFontDict(name) { + class PostScriptParser { constructor(lexer) { this.lexer = lexer; @@ -26366,6 +31235,13 @@ class PostScriptLexer { ;// CONCATENATED MODULE: ./src/core/image_utils.js + + + + + + + class BaseLocalCache { constructor(options) { if (this.constructor === BaseLocalCache) { @@ -26602,6 +31478,14 @@ class GlobalImageCache { + + + + + + + + class PDFFunctionFactory { constructor({ xref, @@ -27865,6 +32749,16 @@ function bidi(str, startLevel = -1, vertical = false) { + + + + + + + + + + const NORMAL = { style: "normal", weight: "normal" @@ -28128,6 +33022,12 @@ function getFontSubstitution(systemFontCache, idFactory, localFontPath, baseFont ;// CONCATENATED MODULE: ./src/core/image_resizer.js + + + + + + const MIN_IMAGE_DIM = 2048; const MAX_IMAGE_DIM = 65537; const MAX_ERROR = 128; @@ -28371,6 +33271,12 @@ class ImageResizer { ImageResizer._goodSquareLength = MIN_IMAGE_DIM; ;// CONCATENATED MODULE: ./src/shared/murmurhash3.js + + + + + + const SEED = 0xc3d2e1f0; const MASK_HIGH = 0xffff0000; const MASK_LOW = 0xffff; @@ -28465,6 +33371,20 @@ class MurmurHash3_64 { ;// CONCATENATED MODULE: ./src/core/operator_list.js + + + + + + + + + + + + + + function addState(parentState, pattern, checkFn, iterateFn, processFn) { let state = parentState; for (let i = 0, ii = pattern.length - 1; i < ii; i++) { @@ -28997,6 +33917,12 @@ class OperatorList { + + + + + + function decodeAndClamp(value, addend, coefficient, max) { value = addend + value * coefficient; if (value < 0) { @@ -29772,6 +34698,24 @@ class PDFImage { + + + + + + + + + + + + + + + + + + @@ -31153,9 +36097,7 @@ class PartialEvaluator { resources, localColorSpaceCache }).then(function (colorSpace) { - if (colorSpace) { - stateManager.state.fillColorSpace = colorSpace; - } + stateManager.state.fillColorSpace = colorSpace || ColorSpace.singletons.gray; })); return; } @@ -31171,9 +36113,7 @@ class PartialEvaluator { resources, localColorSpaceCache }).then(function (colorSpace) { - if (colorSpace) { - stateManager.state.strokeColorSpace = colorSpace; - } + stateManager.state.strokeColorSpace = colorSpace || ColorSpace.singletons.gray; })); return; } @@ -31216,7 +36156,12 @@ class PartialEvaluator { args = ColorSpace.singletons.rgb.getRgb(args, 0); break; case OPS.setFillColorN: - cs = stateManager.state.fillColorSpace; + cs = stateManager.state.patternFillColorSpace; + if (!cs) { + args = []; + fn = OPS.setFillTransparent; + break; + } if (cs.name === "Pattern") { next(self.handleColorN(operatorList, OPS.setFillColorN, args, cs, patterns, resources, task, localColorSpaceCache, localTilingPatternCache, localShadingPatternCache)); return; @@ -31225,7 +36170,12 @@ class PartialEvaluator { fn = OPS.setFillRGBColor; break; case OPS.setStrokeColorN: - cs = stateManager.state.strokeColorSpace; + cs = stateManager.state.patternStrokeColorSpace; + if (!cs) { + args = []; + fn = OPS.setStrokeTransparent; + break; + } if (cs.name === "Pattern") { next(self.handleColorN(operatorList, OPS.setStrokeColorN, args, cs, patterns, resources, task, localColorSpaceCache, localTilingPatternCache, localShadingPatternCache)); return; @@ -32440,6 +37390,9 @@ class PartialEvaluator { map[charCode] = String.fromCodePoint(token); return; } + if (token.length % 2 !== 0) { + token = "\u0000" + token; + } const str = []; for (let k = 0; k < token.length; k += 2) { const w1 = token.charCodeAt(k) << 8 | token.charCodeAt(k + 1); @@ -33238,8 +38191,22 @@ class EvalState { this.ctm = new Float32Array(IDENTITY_MATRIX); this.font = null; this.textRenderingMode = TextRenderingMode.FILL; - this.fillColorSpace = ColorSpace.singletons.gray; - this.strokeColorSpace = ColorSpace.singletons.gray; + this._fillColorSpace = ColorSpace.singletons.gray; + this._strokeColorSpace = ColorSpace.singletons.gray; + this.patternFillColorSpace = null; + this.patternStrokeColorSpace = null; + } + get fillColorSpace() { + return this._fillColorSpace; + } + set fillColorSpace(colorSpace) { + this._fillColorSpace = this.patternFillColorSpace = colorSpace; + } + get strokeColorSpace() { + return this._strokeColorSpace; + } + set strokeColorSpace(colorSpace) { + this._strokeColorSpace = this.patternStrokeColorSpace = colorSpace; } clone() { return Object.create(this); @@ -33729,6 +38696,13 @@ class EvaluatorPreprocessor { + + + + + + + class DefaultAppearanceEvaluator extends EvaluatorPreprocessor { constructor(str) { super(new StringStream(str)); @@ -34112,6 +39086,7 @@ class FakeUnicodeFont { ;// CONCATENATED MODULE: ./src/core/name_number_tree.js + class NameOrNumberTree { constructor(root, xref, type) { if (this.constructor === NameOrNumberTree) { @@ -34330,6 +39305,8 @@ class FileSpec { ;// CONCATENATED MODULE: ./src/core/xml_parser.js + + const XMLParserErrorCode = { NoError: 0, EndOfDocument: -1, @@ -34748,6 +39725,10 @@ class SimpleXMLParser extends XMLParserBase { ;// CONCATENATED MODULE: ./src/core/metadata_parser.js + + + + class MetadataParser { constructor(data) { data = this._repair(data); @@ -34884,6 +39865,15 @@ class DecryptStream extends DecodeStream { + + + + + + + + + class ARCFourCipher { constructor(key) { this.a = 0; @@ -36158,6 +41148,16 @@ class CipherTransformFactory { + + + + + + + + + + async function writeObject(ref, obj, buffer, { encrypt = null }) { @@ -36550,6 +41550,10 @@ async function incrementalUpdate({ + + + + const MAX_DEPTH = 40; const StructElementType = { PAGE_CONTENT: 1, @@ -36834,19 +41838,19 @@ class StructTreeRoot { const tagDict = new Dict(xref); tagDict.set("S", Name.get(type)); if (title) { - tagDict.set("T", title); + tagDict.set("T", stringToAsciiOrUTF16BE(title)); } if (lang) { tagDict.set("Lang", lang); } if (alt) { - tagDict.set("Alt", alt); + tagDict.set("Alt", stringToAsciiOrUTF16BE(alt)); } if (expanded) { - tagDict.set("E", expanded); + tagDict.set("E", stringToAsciiOrUTF16BE(expanded)); } if (actualText) { - tagDict.set("ActualText", actualText); + tagDict.set("ActualText", stringToAsciiOrUTF16BE(actualText)); } await this.#updateParentTag({ structTreeParent, @@ -37261,6 +42265,17 @@ class StructTreePage { + + + + + + + + + + + function isValidExplicitDest(dest) { @@ -37274,26 +42289,27 @@ function isValidExplicitDest(dest) { if (!(zoom instanceof Name)) { return false; } + const argsLen = args.length; let allowNull = true; switch (zoom.name) { case "XYZ": - if (args.length !== 3) { + if (argsLen < 2 || argsLen > 3) { return false; } break; case "Fit": case "FitB": - return args.length === 0; + return argsLen === 0; case "FitH": case "FitBH": case "FitV": case "FitBV": - if (args.length !== 1) { + if (argsLen > 1) { return false; } break; case "FitR": - if (args.length !== 4) { + if (argsLen !== 4) { return false; } allowNull = false; @@ -38684,6 +43700,7 @@ class Catalog { + function mayHaveChildren(value) { return value instanceof Ref || value instanceof Dict || value instanceof BaseStream || Array.isArray(value); } @@ -38926,6 +43943,10 @@ const NamespaceIds = { ;// CONCATENATED MODULE: ./src/core/xfa/utils.js + + + + const dimConverters = { pt: x => x, cm: x => x / 2.54 * 72, @@ -39125,6 +44146,16 @@ class HTMLResult { + + + + + + + + + + class FontFinder { constructor(pdfFonts) { this.fonts = new Map(); @@ -39276,6 +44307,9 @@ function fonts_getMetrics(xfaFont, real = false) { ;// CONCATENATED MODULE: ./src/core/xfa/text.js + + + const WIDTH_FACTOR = 1.02; class FontInfo { constructor(xfaFont, margin, lineHeight, fontFinder) { @@ -39487,6 +44521,10 @@ class TextMeasure { ;// CONCATENATED MODULE: ./src/core/xfa/som.js + + + + const namePattern = /^[^.[]+/; const indexPattern = /^[^\]]+/; const operators = { @@ -39724,6 +44762,17 @@ function createDataNode(root, container, expr) { + + + + + + + + + + + const _applyPrototype = Symbol(); const _attributes = Symbol(); const _attributeNames = Symbol(); @@ -40531,6 +45580,10 @@ class Option10 extends IntegerObject { + + + + function measureToString(m) { if (typeof m === "string") { return "0px"; @@ -41031,6 +46084,9 @@ function fixURL(str) { ;// CONCATENATED MODULE: ./src/core/xfa/layout.js + + + function createLine(node, children) { return { name: "div", @@ -41306,6 +46362,26 @@ function checkDimensions(node, space) { + + + + + + + + + + + + + + + + + + + + const TEMPLATE_NS_ID = NamespaceIds.template.id; const SVG_NS = "http://www.w3.org/2000/svg"; @@ -46180,6 +51256,11 @@ class TemplateNamespace { + + + + + const bind_NS_DATASETS = NamespaceIds.datasets.id; function createText(content) { const node = new Text({}); @@ -46596,6 +51677,7 @@ class Binder { ;// CONCATENATED MODULE: ./src/core/xfa/data.js + class DataHandler { constructor(root, data) { this.data = data; @@ -46649,6 +51731,12 @@ class DataHandler { + + + + + + const CONFIG_NS_ID = NamespaceIds.config.id; class Acrobat extends XFAObject { constructor(attributes) { @@ -48603,6 +53691,16 @@ class XdpNamespace { + + + + + + + + + + const XHTML_NS_ID = NamespaceIds.xhtml.id; const $richText = Symbol(); const VALID_STYLES = new Set(["color", "font", "font-family", "font-size", "font-stretch", "font-style", "font-weight", "margin", "margin-bottom", "margin-left", "margin-right", "margin-top", "letter-spacing", "line-height", "orphans", "page-break-after", "page-break-before", "page-break-inside", "tab-interval", "tab-stop", "text-align", "text-decoration", "text-indent", "vertical-align", "widows", "kerning-mode", "xfa-font-horizontal-scale", "xfa-font-vertical-scale", "xfa-spacerun", "xfa-tab-stops"]); @@ -49038,6 +54136,10 @@ class UnknownNamespace { + + + + class Root extends XFAObject { constructor(ids) { super(-1, "root", Object.create(null)); @@ -49203,6 +54305,14 @@ class Builder { + + + + + + + + class XFAParser extends XMLParserBase { constructor(rootNameSpace = null, richText = false) { super(); @@ -49339,6 +54449,11 @@ class XFAParser extends XMLParserBase { + + + + + class XFAFactory { constructor(data) { try { @@ -49481,6 +54596,21 @@ class XFAFactory { + + + + + + + + + + + + + + + @@ -49892,7 +55022,8 @@ class Annotation { subtype: params.subtype, hasOwnCanvas: false, noRotate: !!(this.flags & AnnotationFlag.NOROTATE), - noHTML: isLocked && isContentLocked + noHTML: isLocked && isContentLocked, + isEditable: false }; if (params.collectFields) { const kids = dict.get("Kids"); @@ -49938,6 +55069,9 @@ class Annotation { } return this.printable; } + mustBeViewedWhenEditing(isEditing, modifiedIds = null) { + return isEditing ? !this.data.isEditable : !modifiedIds?.has(this.data.id); + } get viewable() { if (this.data.quadPoints === null) { return false; @@ -50119,7 +55253,7 @@ class Annotation { }); }); } - async getOperatorList(evaluator, task, intent, renderForms, annotationStorage) { + async getOperatorList(evaluator, task, intent, annotationStorage) { const { hasOwnCanvas, id, @@ -50501,14 +55635,21 @@ class MarkupAnnotation extends Annotation { this._streams.push(this.appearance, appearanceStream); } static async createNewAnnotation(xref, annotation, dependencies, params) { - const annotationRef = annotation.ref ||= xref.getNewTemporaryRef(); + let oldAnnotation; + if (annotation.ref) { + oldAnnotation = (await xref.fetchIfRefAsync(annotation.ref)).clone(); + } else { + annotation.ref = xref.getNewTemporaryRef(); + } + const annotationRef = annotation.ref; const ap = await this.createNewAppearanceStream(annotation, xref, params); const buffer = []; let annotationDict; if (ap) { const apRef = xref.getNewTemporaryRef(); annotationDict = this.createNewDict(annotation, xref, { - apRef + apRef, + oldAnnotation }); await writeObject(apRef, ap, buffer, xref); dependencies.push({ @@ -50516,7 +55657,9 @@ class MarkupAnnotation extends Annotation { data: buffer.join("") }); } else { - annotationDict = this.createNewDict(annotation, xref, {}); + annotationDict = this.createNewDict(annotation, xref, { + oldAnnotation + }); } if (Number.isInteger(annotation.parentTreeId)) { annotationDict.set("StructParent", annotation.parentTreeId); @@ -50674,8 +55817,8 @@ class WidgetAnnotation extends Annotation { } return str; } - async getOperatorList(evaluator, task, intent, renderForms, annotationStorage) { - if (renderForms && !(this instanceof SignatureWidgetAnnotation) && !this.data.noHTML && !this.data.hasOwnCanvas) { + async getOperatorList(evaluator, task, intent, annotationStorage) { + if (intent & RenderingIntentFlag.ANNOTATIONS_FORMS && !(this instanceof SignatureWidgetAnnotation) && !this.data.noHTML && !this.data.hasOwnCanvas) { return { opList: new OperatorList(), separateForm: true, @@ -50683,11 +55826,11 @@ class WidgetAnnotation extends Annotation { }; } if (!this._hasText) { - return super.getOperatorList(evaluator, task, intent, renderForms, annotationStorage); + return super.getOperatorList(evaluator, task, intent, annotationStorage); } const content = await this._getAppearance(evaluator, task, intent, annotationStorage); if (this.appearance && content === null) { - return super.getOperatorList(evaluator, task, intent, renderForms, annotationStorage); + return super.getOperatorList(evaluator, task, intent, annotationStorage); } const opList = new OperatorList(); if (!this._defaultAppearance || content === null) { @@ -50785,8 +55928,7 @@ class WidgetAnnotation extends Annotation { path: this.data.fieldName, value }; - const encoder = val => isAscii(val) ? val : stringToUTF16String(val, true); - dict.set("V", Array.isArray(value) ? value.map(encoder) : encoder(value)); + dict.set("V", Array.isArray(value) ? value.map(stringToAsciiOrUTF16BE) : stringToAsciiOrUTF16BE(value)); this.amendSavedDict(annotationStorage, dict); const maybeMK = this._getMKDict(rotation); if (maybeMK) { @@ -51253,7 +56395,7 @@ class ButtonWidgetAnnotation extends WidgetAnnotation { warn("Invalid field flags for button widget annotation"); } } - async getOperatorList(evaluator, task, intent, renderForms, annotationStorage) { + async getOperatorList(evaluator, task, intent, annotationStorage) { if (this.data.pushButton) { return super.getOperatorList(evaluator, task, intent, false, annotationStorage); } @@ -51265,7 +56407,7 @@ class ButtonWidgetAnnotation extends WidgetAnnotation { rotation = storageEntry ? storageEntry.rotation : null; } if (value === null && this.appearance) { - return super.getOperatorList(evaluator, task, intent, renderForms, annotationStorage); + return super.getOperatorList(evaluator, task, intent, annotationStorage); } if (value === null || value === undefined) { value = this.data.checkBox ? this.data.fieldValue === this.data.exportValue : this.data.fieldValue === this.data.buttonValue; @@ -51278,7 +56420,7 @@ class ButtonWidgetAnnotation extends WidgetAnnotation { appearance.dict.set("Matrix", this.getRotationMatrix(annotationStorage)); } this.appearance = appearance; - const operatorList = super.getOperatorList(evaluator, task, intent, renderForms, annotationStorage); + const operatorList = super.getOperatorList(evaluator, task, intent, annotationStorage); this.appearance = savedAppearance; appearance.dict.set("Matrix", savedMatrix); return operatorList; @@ -51895,7 +57037,8 @@ class PopupAnnotation extends Annotation { class FreeTextAnnotation extends MarkupAnnotation { constructor(params) { super(params); - this.data.hasOwnCanvas = !this.data.noHTML; + this.data.hasOwnCanvas = this.data.noRotate; + this.data.isEditable = !this.data.noHTML; this.data.noHTML = false; const { evaluatorOptions, @@ -51941,7 +57084,8 @@ class FreeTextAnnotation extends MarkupAnnotation { } static createNewDict(annotation, xref, { apRef, - ap + ap, + oldAnnotation }) { const { color, @@ -51951,19 +57095,24 @@ class FreeTextAnnotation extends MarkupAnnotation { user, value } = annotation; - const freetext = new Dict(xref); + const freetext = oldAnnotation || new Dict(xref); freetext.set("Type", Name.get("Annot")); freetext.set("Subtype", Name.get("FreeText")); - freetext.set("CreationDate", `D:${getModificationDate()}`); + if (oldAnnotation) { + freetext.set("M", `D:${getModificationDate()}`); + freetext.delete("RC"); + } else { + freetext.set("CreationDate", `D:${getModificationDate()}`); + } freetext.set("Rect", rect); const da = `/Helv ${fontSize} Tf ${getPdfColor(color, true)}`; freetext.set("DA", da); - freetext.set("Contents", isAscii(value) ? value : stringToUTF16String(value, true)); + freetext.set("Contents", stringToAsciiOrUTF16BE(value)); freetext.set("F", 4); freetext.set("Border", [0, 0, 0]); freetext.set("Rotate", rotation); if (user) { - freetext.set("T", isAscii(user) ? user : stringToUTF16String(user, true)); + freetext.set("T", stringToAsciiOrUTF16BE(user)); } if (apRef || ap) { const n = new Dict(xref); @@ -52534,7 +57683,7 @@ class HighlightAnnotation extends MarkupAnnotation { highlight.set("C", Array.from(color, c => c / 255)); highlight.set("CA", opacity); if (user) { - highlight.set("T", isAscii(user) ? user : stringToUTF16String(user, true)); + highlight.set("T", stringToAsciiOrUTF16BE(user)); } if (apRef || ap) { const n = new Dict(xref); @@ -52770,7 +57919,7 @@ class StampAnnotation extends MarkupAnnotation { stamp.set("Border", [0, 0, 0]); stamp.set("Rotate", rotation); if (user) { - stamp.set("T", isAscii(user) ? user : stringToUTF16String(user, true)); + stamp.set("T", stringToAsciiOrUTF16BE(user)); } if (apRef || ap) { const n = new Dict(xref); @@ -52835,6 +57984,7 @@ class FileAttachmentAnnotation extends MarkupAnnotation { + function decodeString(str) { try { return stringToUTF8String(str); @@ -52895,6 +58045,20 @@ class DatasetReader { + + + + + + + + + + + + + + class XRef { #firstXRefStmPos = null; constructor(stream, pdfManager) { @@ -53633,6 +58797,25 @@ class XRef { + + + + + + + + + + + + + + + + + + + const DEFAULT_USER_UNIT = 1.0; @@ -53887,7 +59070,8 @@ class Page { task, intent, cacheKey, - annotationStorage = null + annotationStorage = null, + modifiedIds = null }) { const contentStreamPromise = this.getContentStream(); const resourcesPromise = this.loadResources(["ColorSpace", "ExtGState", "Font", "Pattern", "Properties", "Shading", "XObject"]); @@ -53984,13 +59168,14 @@ class Page { }; } const renderForms = !!(intent & RenderingIntentFlag.ANNOTATIONS_FORMS), + isEditing = !!(intent & RenderingIntentFlag.IS_EDITING), intentAny = !!(intent & RenderingIntentFlag.ANY), intentDisplay = !!(intent & RenderingIntentFlag.DISPLAY), intentPrint = !!(intent & RenderingIntentFlag.PRINT); const opListPromises = []; for (const annotation of annotations) { - if (intentAny || intentDisplay && annotation.mustBeViewed(annotationStorage, renderForms) || intentPrint && annotation.mustBePrinted(annotationStorage)) { - opListPromises.push(annotation.getOperatorList(partialEvaluator, task, intent, renderForms, annotationStorage).catch(function (reason) { + if (intentAny || intentDisplay && annotation.mustBeViewed(annotationStorage, renderForms) && annotation.mustBeViewedWhenEditing(isEditing, modifiedIds) || intentPrint && annotation.mustBePrinted(annotationStorage)) { + opListPromises.push(annotation.getOperatorList(partialEvaluator, task, intent, annotationStorage).catch(function (reason) { warn("getOperatorList - ignoring annotation data during " + `"${task.name}" task: "${reason}".`); return { opList: null, @@ -55143,6 +60328,7 @@ class NetworkPdfManager extends BasePdfManager { ;// CONCATENATED MODULE: ./src/shared/message_handler.js + const CallbackKind = { UNKNOWN: 0, DATA: 1, @@ -55546,6 +60732,7 @@ class MessageHandler { ;// CONCATENATED MODULE: ./src/core/worker_stream.js + class PDFWorkerStream { constructor(msgHandler) { this._msgHandler = msgHandler; @@ -55661,6 +60848,26 @@ class PDFWorkerStreamRangeReader { + + + + + + + + + + + + + + + + + + + + class WorkerTask { constructor(name) { @@ -55710,7 +60917,7 @@ class WorkerMessageHandler { docId, apiVersion } = docParams; - const workerVersion = "4.4.168"; + const workerVersion = "4.5.136"; if (apiVersion !== workerVersion) { throw new Error(`The API version "${apiVersion}" does not match ` + `the Worker version "${workerVersion}".`); } @@ -56178,7 +61385,8 @@ class WorkerMessageHandler { task, intent: data.intent, cacheKey: data.cacheKey, - annotationStorage: data.annotationStorage + annotationStorage: data.annotationStorage, + modifiedIds: data.modifiedIds }).then(function (operatorListInfo) { finishWorkerTask(task); if (start) { @@ -56280,8 +61488,8 @@ if (typeof window === "undefined" && !isNodeJS && typeof self !== "undefined" && ;// CONCATENATED MODULE: ./src/pdf.worker.js -const pdfjsVersion = "4.4.168"; -const pdfjsBuild = "19fbc8998"; +const pdfjsVersion = "4.5.136"; +const pdfjsBuild = "3a21f03b0"; var __webpack_exports__WorkerMessageHandler = __webpack_exports__.WorkerMessageHandler; export { __webpack_exports__WorkerMessageHandler as WorkerMessageHandler }; diff --git a/cps/static/js/libs/plugins.js b/cps/static/js/libs/plugins.js index ee7ded31..9190ea2f 100644 --- a/cps/static/js/libs/plugins.js +++ b/cps/static/js/libs/plugins.js @@ -39,7 +39,7 @@ !function(t,e){"object"==typeof module&&module.exports?module.exports=e(t,require("jquery")):t.jQueryBridget=e(t,t.jQuery)}(window,(function(t,e){let i=t.console,n=void 0===i?function(){}:function(t){i.error(t)};return function(i,o,s){(s=s||e||t.jQuery)&&(o.prototype.option||(o.prototype.option=function(t){t&&(this.options=Object.assign(this.options||{},t))}),s.fn[i]=function(t,...e){return"string"==typeof t?function(t,e,o){let r,l=`$().${i}("${e}")`;return t.each((function(t,h){let a=s.data(h,i);if(!a)return void n(`${i} not initialized. Cannot call method ${l}`);let c=a[e];if(!c||"_"==e.charAt(0))return void n(`${l} is not a valid method`);let u=c.apply(a,o);r=void 0===r?u:r})),void 0!==r?r:t}(this,t,e):(r=t,this.each((function(t,e){let n=s.data(e,i);n?(n.option(r),n._init()):(n=new o(e,r),s.data(e,i,n))})),this);var r})}})),function(t,e){"object"==typeof module&&module.exports?module.exports=e():t.EvEmitter=e()}("undefined"!=typeof window?window:this,(function(){function t(){}let e=t.prototype;return e.on=function(t,e){if(!t||!e)return this;let i=this._events=this._events||{},n=i[t]=i[t]||[];return n.includes(e)||n.push(e),this},e.once=function(t,e){if(!t||!e)return this;this.on(t,e);let i=this._onceEvents=this._onceEvents||{};return(i[t]=i[t]||{})[e]=!0,this},e.off=function(t,e){let i=this._events&&this._events[t];if(!i||!i.length)return this;let n=i.indexOf(e);return-1!=n&&i.splice(n,1),this},e.emitEvent=function(t,e){let i=this._events&&this._events[t];if(!i||!i.length)return this;i=i.slice(0),e=e||[];let n=this._onceEvents&&this._onceEvents[t];for(let o of i){n&&n[o]&&(this.off(t,o),delete n[o]),o.apply(this,e)}return this},e.allOff=function(){return delete this._events,delete this._onceEvents,this},t})),function(t,e){"object"==typeof module&&module.exports?module.exports=e(t):t.fizzyUIUtils=e(t)}(this,(function(t){let e={extend:function(t,e){return Object.assign(t,e)},modulo:function(t,e){return(t%e+e)%e},makeArray:function(t){if(Array.isArray(t))return t;if(null==t)return[];return"object"==typeof t&&"number"==typeof t.length?[...t]:[t]},removeFrom:function(t,e){let i=t.indexOf(e);-1!=i&&t.splice(i,1)},getParent:function(t,e){for(;t.parentNode&&t!=document.body;)if((t=t.parentNode).matches(e))return t},getQueryElement:function(t){return"string"==typeof t?document.querySelector(t):t},handleEvent:function(t){let e="on"+t.type;this[e]&&this[e](t)},filterFindElements:function(t,i){return(t=e.makeArray(t)).filter((t=>t instanceof HTMLElement)).reduce(((t,e)=>{if(!i)return t.push(e),t;e.matches(i)&&t.push(e);let n=e.querySelectorAll(i);return t=t.concat(...n)}),[])},debounceMethod:function(t,e,i){i=i||100;let n=t.prototype[e],o=e+"Timeout";t.prototype[e]=function(){clearTimeout(this[o]);let t=arguments;this[o]=setTimeout((()=>{n.apply(this,t),delete this[o]}),i)}},docReady:function(t){let e=document.readyState;"complete"==e||"interactive"==e?setTimeout(t):document.addEventListener("DOMContentLoaded",t)},toDashed:function(t){return t.replace(/(.)([A-Z])/g,(function(t,e,i){return e+"-"+i})).toLowerCase()}},i=t.console;return e.htmlInit=function(n,o){e.docReady((function(){let s="data-"+e.toDashed(o),r=document.querySelectorAll(`[${s}]`),l=t.jQuery;[...r].forEach((t=>{let e,r=t.getAttribute(s);try{e=r&&JSON.parse(r)}catch(e){return void(i&&i.error(`Error parsing ${s} on ${t.className}: ${e}`))}let h=new n(t,e);l&&l.data(t,o,h)}))}))},e})),function(t,e){"object"==typeof module&&module.exports?module.exports=e(t,require("ev-emitter"),require("fizzy-ui-utils")):t.InfiniteScroll=e(t,t.EvEmitter,t.fizzyUIUtils)}(window,(function(t,e,i){let n=t.jQuery,o={};function s(t,e){let r=i.getQueryElement(t);if(r){if((t=r).infiniteScrollGUID){let i=o[t.infiniteScrollGUID];return i.option(e),i}this.element=t,this.options={...s.defaults},this.option(e),n&&(this.$element=n(this.element)),this.create()}else console.error("Bad element for InfiniteScroll: "+(r||t))}s.defaults={},s.create={},s.destroy={};let r=s.prototype;Object.assign(r,e.prototype);let l=0;r.create=function(){let t=this.guid=++l;if(this.element.infiniteScrollGUID=t,o[t]=this,this.pageIndex=1,this.loadCount=0,this.updateGetPath(),this.getPath&&this.getPath()){this.updateGetAbsolutePath(),this.log("initialized",[this.element.className]),this.callOnInit();for(let t in s.create)s.create[t].call(this)}else console.error("Disabling InfiniteScroll")},r.option=function(t){Object.assign(this.options,t)},r.callOnInit=function(){let t=this.options.onInit;t&&t.call(this,this)},r.dispatchEvent=function(t,e,i){this.log(t,i);let o=e?[e].concat(i):i;if(this.emitEvent(t,o),!n||!this.$element)return;let s=t+=".infiniteScroll";if(e){let i=n.Event(e);i.type=t,s=i}this.$element.trigger(s,i)};let h={initialized:t=>`on ${t}`,request:t=>`URL: ${t}`,load:(t,e)=>`${t.title||""}. URL: ${e}`,error:(t,e)=>`${t}. URL: ${e}`,append:(t,e,i)=>`${i.length} items. URL: ${e}`,last:(t,e)=>`URL: ${e}`,history:(t,e)=>`URL: ${e}`,pageIndex:function(t,e){return`current page determined to be: ${t} from ${e}`}};r.log=function(t,e){if(!this.options.debug)return;let i=`[InfiniteScroll] ${t}`,n=h[t];n&&(i+=". "+n.apply(this,e)),console.log(i)},r.updateMeasurements=function(){this.windowHeight=t.innerHeight;let e=this.element.getBoundingClientRect();this.top=e.top+t.scrollY},r.updateScroller=function(){let e=this.options.elementScroll;if(e){if(this.scroller=!0===e?this.element:i.getQueryElement(e),!this.scroller)throw new Error(`Unable to find elementScroll: ${e}`)}else this.scroller=t},r.updateGetPath=function(){let t=this.options.path;if(!t)return void console.error(`InfiniteScroll path option required. Set as: ${t}`);let e=typeof t;"function"!=e?"string"==e&&t.match("{{#}}")?this.updateGetPathTemplate(t):this.updateGetPathSelector(t):this.getPath=t},r.updateGetPathTemplate=function(t){this.getPath=()=>{let e=this.pageIndex+1;return t.replace("{{#}}",e)};let e=t.replace(/(\\\?|\?)/,"\\?").replace("{{#}}","(\\d\\d?\\d?)"),i=new RegExp(e),n=location.href.match(i);n&&(this.pageIndex=parseInt(n[1],10),this.log("pageIndex",[this.pageIndex,"template string"]))};let a=[/^(.*?\/?page\/?)(\d\d?\d?)(.*?$)/,/^(.*?\/?\?page=)(\d\d?\d?)(.*?$)/,/(.*?)(\d\d?\d?)(?!.*\d)(.*?$)/],c=s.getPathParts=function(t){if(t)for(let e of a){let i=t.match(e);if(i){let[,t,e,n]=i;return{begin:t,index:e,end:n}}}};r.updateGetPathSelector=function(t){let e=document.querySelector(t);if(!e)return void console.error(`Bad InfiniteScroll path option. Next link not found: ${t}`);let i=e.getAttribute("href"),n=c(i);if(!n)return void console.error(`InfiniteScroll unable to parse next link href: ${i}`);let{begin:o,index:s,end:r}=n;this.isPathSelector=!0,this.getPath=()=>o+(this.pageIndex+1)+r,this.pageIndex=parseInt(s,10)-1,this.log("pageIndex",[this.pageIndex,"next link"])},r.updateGetAbsolutePath=function(){let t=this.getPath();if(t.match(/^http/)||t.match(/^\//))return void(this.getAbsolutePath=this.getPath);let{pathname:e}=location,i=t.match(/^\?/),n=e.substring(0,e.lastIndexOf("/")),o=i?e:n+"/";this.getAbsolutePath=()=>o+this.getPath()},s.create.hideNav=function(){let t=i.getQueryElement(this.options.hideNav);t&&(t.style.display="none",this.nav=t)},s.destroy.hideNav=function(){this.nav&&(this.nav.style.display="")},r.destroy=function(){this.allOff();for(let t in s.destroy)s.destroy[t].call(this);delete this.element.infiniteScrollGUID,delete o[this.guid],n&&this.$element&&n.removeData(this.element,"infiniteScroll")},s.throttle=function(t,e){let i,n;return e=e||200,function(){let o=+new Date,s=arguments,r=()=>{i=o,t.apply(this,s)};i&&o{if(!i.ok){let t=new Error(i.statusText);return this.onPageError(t,o,i),{response:i}}return i[t]().then((s=>("text"==t&&e&&(s=n.parseFromString(s,"text/html")),204==i.status?(this.lastPageReached(s,o),{body:s,response:i}):this.onPageLoad(s,o,i))))})).catch((t=>{this.onPageError(t,o)}));return this.dispatchEvent("request",null,[o,s]),s},i.onPageLoad=function(t,e,i){return this.options.append||(this.isLoading=!1),this.pageIndex++,this.loadCount++,this.dispatchEvent("load",null,[t,e,i]),this.appendNextPage(t,e,i)},i.appendNextPage=function(t,e,i){let{append:n,responseBody:s,domParseResponse:r}=this.options;if(!("text"==s&&r)||!n)return{body:t,response:i};let l=t.querySelectorAll(n),h={body:t,response:i,items:l};if(!l||!l.length)return this.lastPageReached(t,e),h;let a=o(l),c=()=>(this.appendItems(l,a),this.isLoading=!1,this.dispatchEvent("append",null,[t,e,l,i]),h);return this.options.outlayer?this.appendOutlayerItems(a,c):c()},i.appendItems=function(t,e){t&&t.length&&(function(t){let e=t.querySelectorAll("script");for(let t of e){let e=document.createElement("script"),i=t.attributes;for(let t of i)e.setAttribute(t.name,t.value);e.innerHTML=t.innerHTML,t.parentNode.replaceChild(e,t)}}(e=e||o(t)),this.element.appendChild(e))},i.appendOutlayerItems=function(i,n){let o=e.imagesLoaded||t.imagesLoaded;return o?new Promise((function(t){o(i,(function(){let e=n();t(e)}))})):(console.error("[InfiniteScroll] imagesLoaded required for outlayer option"),void(this.isLoading=!1))},i.onAppendOutlayer=function(t,e,i){this.options.outlayer.appended(i)},i.checkLastPage=function(t,e){let i,{checkLastPage:n,path:o}=this.options;if(n){if("function"==typeof o){if(!this.getPath())return void this.lastPageReached(t,e)}"string"==typeof n?i=n:this.isPathSelector&&(i=o),i&&t.querySelector&&(t.querySelector(i)||this.lastPageReached(t,e))}},i.lastPageReached=function(t,e){this.canLoad=!1,this.dispatchEvent("last",null,[t,e])},i.onPageError=function(t,e,i){return this.isLoading=!1,this.canLoad=!1,this.dispatchEvent("error",null,[t,e,i]),t},e.create.prefill=function(){if(!this.options.prefill)return;let t=this.options.append;t?(this.updateMeasurements(),this.updateScroller(),this.isPrefilling=!0,this.on("append",this.prefill),this.once("error",this.stopPrefill),this.once("last",this.stopPrefill),this.prefill()):console.error(`append option required for prefill. Set as :${t}`)},i.prefill=function(){let t=this.getPrefillDistance();this.isPrefilling=t>=0,this.isPrefilling?(this.log("prefill"),this.loadNextPage()):this.stopPrefill()},i.getPrefillDistance=function(){return this.options.elementScroll?this.scroller.clientHeight-this.scroller.scrollHeight:this.windowHeight-this.element.clientHeight},i.stopPrefill=function(){this.log("stopPrefill"),this.off("append",this.prefill)},e})),function(t,e){"object"==typeof module&&module.exports?module.exports=e(t,require("./core"),require("fizzy-ui-utils")):e(t,t.InfiniteScroll,t.fizzyUIUtils)}(window,(function(t,e,i){let n=e.prototype;return Object.assign(e.defaults,{scrollThreshold:400}),e.create.scrollWatch=function(){this.pageScrollHandler=this.onPageScroll.bind(this),this.resizeHandler=this.onResize.bind(this);let t=this.options.scrollThreshold;(t||0===t)&&this.enableScrollWatch()},e.destroy.scrollWatch=function(){this.disableScrollWatch()},n.enableScrollWatch=function(){this.isScrollWatching||(this.isScrollWatching=!0,this.updateMeasurements(),this.updateScroller(),this.on("last",this.disableScrollWatch),this.bindScrollWatchEvents(!0))},n.disableScrollWatch=function(){this.isScrollWatching&&(this.bindScrollWatchEvents(!1),delete this.isScrollWatching)},n.bindScrollWatchEvents=function(e){let i=e?"addEventListener":"removeEventListener";this.scroller[i]("scroll",this.pageScrollHandler),t[i]("resize",this.resizeHandler)},n.onPageScroll=e.throttle((function(){this.getBottomDistance()<=this.options.scrollThreshold&&this.dispatchEvent("scrollThreshold")})),n.getBottomDistance=function(){let e,i;return this.options.elementScroll?(e=this.scroller.scrollHeight,i=this.scroller.scrollTop+this.scroller.clientHeight):(e=this.top+this.element.clientHeight,i=t.scrollY+this.windowHeight),e-i},n.onResize=function(){this.updateMeasurements()},i.debounceMethod(e,"onResize",150),e})),function(t,e){"object"==typeof module&&module.exports?module.exports=e(t,require("./core"),require("fizzy-ui-utils")):e(t,t.InfiniteScroll,t.fizzyUIUtils)}(window,(function(t,e,i){let n=e.prototype;Object.assign(e.defaults,{history:"replace"});let o=document.createElement("a");return e.create.history=function(){if(!this.options.history)return;o.href=this.getAbsolutePath(),(o.origin||o.protocol+"//"+o.host)==location.origin?this.options.append?this.createHistoryAppend():this.createHistoryPageLoad():console.error(`[InfiniteScroll] cannot set history with different origin: ${o.origin} on ${location.origin} . History behavior disabled.`)},n.createHistoryAppend=function(){this.updateMeasurements(),this.updateScroller(),this.scrollPages=[{top:0,path:location.href,title:document.title}],this.scrollPage=this.scrollPages[0],this.scrollHistoryHandler=this.onScrollHistory.bind(this),this.unloadHandler=this.onUnload.bind(this),this.scroller.addEventListener("scroll",this.scrollHistoryHandler),this.on("append",this.onAppendHistory),this.bindHistoryAppendEvents(!0)},n.bindHistoryAppendEvents=function(e){let i=e?"addEventListener":"removeEventListener";this.scroller[i]("scroll",this.scrollHistoryHandler),t[i]("unload",this.unloadHandler)},n.createHistoryPageLoad=function(){this.on("load",this.onPageLoadHistory)},e.destroy.history=n.destroyHistory=function(){this.options.history&&this.options.append&&this.bindHistoryAppendEvents(!1)},n.onAppendHistory=function(t,e,i){if(!i||!i.length)return;let n=i[0],s=this.getElementScrollY(n);o.href=e,this.scrollPages.push({top:s,path:o.href,title:t.title})},n.getElementScrollY=function(e){if(this.options.elementScroll)return e.offsetTop-this.top;return e.getBoundingClientRect().top+t.scrollY},n.onScrollHistory=function(){let t=this.getClosestScrollPage();t!=this.scrollPage&&(this.scrollPage=t,this.setHistory(t.title,t.path))},i.debounceMethod(e,"onScrollHistory",150),n.getClosestScrollPage=function(){let e,i;e=this.options.elementScroll?this.scroller.scrollTop+this.scroller.clientHeight/2:t.scrollY+this.windowHeight/2;for(let t of this.scrollPages){if(t.top>=e)break;i=t}return i},n.setHistory=function(t,e){let i=this.options.history;i&&history[i+"State"]&&(history[i+"State"](null,t,e),this.options.historyTitle&&(document.title=t),this.dispatchEvent("history",null,[t,e]))},n.onUnload=function(){if(0===this.scrollPage.top)return;let e=t.scrollY-this.scrollPage.top+this.top;this.destroyHistory(),scrollTo(0,e)},n.onPageLoadHistory=function(t,e){this.setHistory(t.title,e)},e})),function(t,e){"object"==typeof module&&module.exports?module.exports=e(t,require("./core"),require("fizzy-ui-utils")):e(t,t.InfiniteScroll,t.fizzyUIUtils)}(window,(function(t,e,i){class n{constructor(t,e){this.element=t,this.infScroll=e,this.clickHandler=this.onClick.bind(this),this.element.addEventListener("click",this.clickHandler),e.on("request",this.disable.bind(this)),e.on("load",this.enable.bind(this)),e.on("error",this.hide.bind(this)),e.on("last",this.hide.bind(this))}onClick(t){t.preventDefault(),this.infScroll.loadNextPage()}enable(){this.element.removeAttribute("disabled")}disable(){this.element.disabled="disabled"}hide(){this.element.style.display="none"}destroy(){this.element.removeEventListener("click",this.clickHandler)}}return e.create.button=function(){let t=i.getQueryElement(this.options.button);t&&(this.button=new n(t,this))},e.destroy.button=function(){this.button&&this.button.destroy()},e.Button=n,e})),function(t,e){"object"==typeof module&&module.exports?module.exports=e(t,require("./core"),require("fizzy-ui-utils")):e(t,t.InfiniteScroll,t.fizzyUIUtils)}(window,(function(t,e,i){let n=e.prototype;function o(t){r(t,"none")}function s(t){r(t,"block")}function r(t,e){t&&(t.style.display=e)}return e.create.status=function(){let t=i.getQueryElement(this.options.status);t&&(this.statusElement=t,this.statusEventElements={request:t.querySelector(".infinite-scroll-request"),error:t.querySelector(".infinite-scroll-error"),last:t.querySelector(".infinite-scroll-last")},this.on("request",this.showRequestStatus),this.on("error",this.showErrorStatus),this.on("last",this.showLastStatus),this.bindHideStatus("on"))},n.bindHideStatus=function(t){let e=this.options.append?"append":"load";this[t](e,this.hideAllStatus)},n.showRequestStatus=function(){this.showStatus("request")},n.showErrorStatus=function(){this.showStatus("error")},n.showLastStatus=function(){this.showStatus("last"),this.bindHideStatus("off")},n.showStatus=function(t){s(this.statusElement),this.hideStatusEventElements(),s(this.statusEventElements[t])},n.hideAllStatus=function(){o(this.statusElement),this.hideStatusEventElements()},n.hideStatusEventElements=function(){for(let t in this.statusEventElements){o(this.statusEventElements[t])}},e})), /*! * imagesLoaded v4.1.4 - * JavaScript is all like "You images are done yet or what?" + * JavaScript is all like "Your images are done yet or what?" * MIT License */ function(t,e){"use strict";"function"==typeof define&&define.amd?define(["ev-emitter/ev-emitter"],(function(i){return e(t,i)})):"object"==typeof module&&module.exports?module.exports=e(t,require("ev-emitter")):t.imagesLoaded=e(t,t.EvEmitter)}("undefined"!=typeof window?window:this,(function(t,e){"use strict";var i=t.jQuery,n=t.console;function o(t,e){for(var i in e)t[i]=e[i];return t}var s=Array.prototype.slice;function r(t,e,l){if(!(this instanceof r))return new r(t,e,l);var h,a=t;("string"==typeof t&&(a=document.querySelectorAll(t)),a)?(this.elements=(h=a,Array.isArray(h)?h:"object"==typeof h&&"number"==typeof h.length?s.call(h):[h]),this.options=o({},this.options),"function"==typeof e?l=e:o(this.options,e),l&&this.on("always",l),this.getImages(),i&&(this.jqDeferred=new i.Deferred),setTimeout(this.check.bind(this))):n.error("Bad element for imagesLoaded "+(a||t))}r.prototype=Object.create(e.prototype),r.prototype.options={},r.prototype.getImages=function(){this.images=[],this.elements.forEach(this.addElementImages,this)},r.prototype.addElementImages=function(t){"IMG"==t.nodeName&&this.addImage(t),!0===this.options.background&&this.addElementBackgroundImages(t);var e=t.nodeType;if(e&&l[e]){for(var i=t.querySelectorAll("img"),n=0;n 1; - (function checkCanvasSizeLimitation() { + (function () { if (isIOS || isAndroid) { - compatibilityParams.maxCanvasPixels = 5242880; + compatParams.set("maxCanvasPixels", 5242880); + } + })(); + (function () { + if (isAndroid) { + compatParams.set("useSystemFonts", false); } })(); } @@ -552,9 +557,21 @@ const OptionKind = { VIEWER: 0x02, API: 0x04, WORKER: 0x08, + EVENT_DISPATCH: 0x10, PREFERENCE: 0x80 }; +const Type = { + BOOLEAN: 0x01, + NUMBER: 0x02, + OBJECT: 0x04, + STRING: 0x08, + UNDEFINED: 0x10 +}; const defaultOptions = { + allowedGlobalEvents: { + value: null, + kind: OptionKind.BROWSER + }, canvasMaxAreaInBytes: { value: -1, kind: OptionKind.BROWSER + OptionKind.API @@ -563,6 +580,16 @@ const defaultOptions = { value: false, kind: OptionKind.BROWSER }, + localeProperties: { + value: { + lang: navigator.language || "en-US" + }, + kind: OptionKind.BROWSER + }, + nimbusDataStr: { + value: "", + kind: OptionKind.BROWSER + }, supportsCaretBrowsingMode: { value: false, kind: OptionKind.BROWSER @@ -587,6 +614,14 @@ const defaultOptions = { value: true, kind: OptionKind.BROWSER }, + toolbarDensity: { + value: 0, + kind: OptionKind.BROWSER + OptionKind.EVENT_DISPATCH + }, + altTextLearnMoreUrl: { + value: "", + kind: OptionKind.VIEWER + OptionKind.PREFERENCE + }, annotationEditorMode: { value: 0, kind: OptionKind.VIEWER + OptionKind.PREFERENCE @@ -619,6 +654,14 @@ const defaultOptions = { value: false, kind: OptionKind.VIEWER + OptionKind.PREFERENCE }, + enableAltText: { + value: false, + kind: OptionKind.VIEWER + OptionKind.PREFERENCE + }, + enableGuessAltText: { + value: true, + kind: OptionKind.VIEWER + OptionKind.PREFERENCE + }, enableHighlightEditor: { value: false, kind: OptionKind.VIEWER + OptionKind.PREFERENCE @@ -627,10 +670,6 @@ const defaultOptions = { value: false, kind: OptionKind.VIEWER + OptionKind.PREFERENCE }, - enableML: { - value: false, - kind: OptionKind.VIEWER + OptionKind.PREFERENCE - }, enablePermissions: { value: false, kind: OptionKind.VIEWER + OptionKind.PREFERENCE @@ -643,8 +682,8 @@ const defaultOptions = { value: true, kind: OptionKind.VIEWER + OptionKind.PREFERENCE }, - enableStampEditor: { - value: true, + enableUpdatedAddImage: { + value: false, kind: OptionKind.VIEWER + OptionKind.PREFERENCE }, externalLinkRel: { @@ -775,6 +814,11 @@ const defaultOptions = { value: "../web/standard_fonts/", kind: OptionKind.API }, + useSystemFonts: { + value: undefined, + kind: OptionKind.API, + type: Type.BOOLEAN + Type.UNDEFINED + }, verbosity: { value: 1, kind: OptionKind.API @@ -784,7 +828,7 @@ const defaultOptions = { kind: OptionKind.WORKER }, workerSrc: { - value: "../build/pdf.worker.mjs", + value: "../build/pdf.worker.js", kind: OptionKind.WORKER } }; @@ -807,62 +851,80 @@ const defaultOptions = { value: false, kind: OptionKind.VIEWER }; - defaultOptions.locale = { - value: navigator.language || "en-US", - kind: OptionKind.VIEWER - }; } -const userOptions = Object.create(null); +const userOptions = new Map(); { - for (const name in compatibilityParams) { - userOptions[name] = compatibilityParams[name]; + for (const [name, value] of compatParams) { + userOptions.set(name, value); } } class AppOptions { + static eventBus; constructor() { throw new Error("Cannot initialize AppOptions."); } static get(name) { - return userOptions[name] ?? defaultOptions[name]?.value ?? undefined; + return userOptions.has(name) ? userOptions.get(name) : defaultOptions[name]?.value; } static getAll(kind = null, defaultOnly = false) { const options = Object.create(null); for (const name in defaultOptions) { - const defaultOption = defaultOptions[name]; - if (kind && !(kind & defaultOption.kind)) { + const defaultOpt = defaultOptions[name]; + if (kind && !(kind & defaultOpt.kind)) { continue; } - options[name] = defaultOnly ? defaultOption.value : userOptions[name] ?? defaultOption.value; + options[name] = !defaultOnly && userOptions.has(name) ? userOptions.get(name) : defaultOpt.value; } return options; } static set(name, value) { - userOptions[name] = value; + this.setAll({ + [name]: value + }); } - static setAll(options, init = false) { - if (init) { - if (this.get("disablePreferences")) { - return; - } - for (const name in userOptions) { - if (compatibilityParams[name] !== undefined) { - continue; - } - console.warn("setAll: The Preferences may override manually set AppOptions; " + 'please use the "disablePreferences"-option in order to prevent that.'); - break; - } - } + static setAll(options, prefs = false) { + let events; for (const name in options) { - userOptions[name] = options[name]; + const defaultOpt = defaultOptions[name], + userOpt = options[name]; + if (!defaultOpt || !(typeof userOpt === typeof defaultOpt.value || Type[(typeof userOpt).toUpperCase()] & defaultOpt.type)) { + continue; + } + const { + kind + } = defaultOpt; + if (prefs && !(kind & OptionKind.BROWSER || kind & OptionKind.PREFERENCE)) { + continue; + } + if (this.eventBus && kind & OptionKind.EVENT_DISPATCH) { + (events ||= new Map()).set(name, userOpt); + } + userOptions.set(name, userOpt); + } + if (events) { + for (const [name, value] of events) { + this.eventBus.dispatch(name.toLowerCase(), { + source: this, + value + }); + } } } - static remove(name) { - delete userOptions[name]; - const val = compatibilityParams[name]; - if (val !== undefined) { - userOptions[name] = val; +} +{ + AppOptions._checkDisablePreferences = () => { + if (AppOptions.get("disablePreferences")) { + return true; } - } + for (const [name] of userOptions) { + if (compatParams.has(name)) { + continue; + } + console.warn("The Preferences may override manually set AppOptions; " + 'please use the "disablePreferences"-option to prevent that.'); + break; + } + return false; + }; } ;// CONCATENATED MODULE: ./web/pdf_link_service.js @@ -1171,26 +1233,27 @@ class PDFLinkService { if (!(typeof zoom === "object" && typeof zoom?.name === "string")) { return false; } + const argsLen = args.length; let allowNull = true; switch (zoom.name) { case "XYZ": - if (args.length !== 3) { + if (argsLen < 2 || argsLen > 3) { return false; } break; case "Fit": case "FitB": - return args.length === 0; + return argsLen === 0; case "FitH": case "FitBH": case "FitV": case "FitBV": - if (args.length !== 1) { + if (argsLen > 1) { return false; } break; case "FitR": - if (args.length !== 4) { + if (argsLen !== 4) { return false; } allowNull = false; @@ -1240,7 +1303,6 @@ const { noContextMenu, normalizeUnicode, OPS, - Outliner, PasswordResponses, PDFDataRangeTransport, PDFDateString, @@ -1248,12 +1310,10 @@ const { PermissionFlag, PixelsPerInch, RenderingCancelledException, - renderTextLayer, setLayerDimensions, shadow, TextLayer, UnexpectedResponseException, - updateTextLayer, Util, VerbosityLevel, version, @@ -1401,40 +1461,28 @@ class BaseExternalServices { updateEditorStates(data) { throw new Error("Not implemented: updateEditorStates"); } - async getNimbusExperimentData() {} - async getGlobalEventNames() { - return null; - } dispatchGlobalEvent(_event) {} } ;// CONCATENATED MODULE: ./web/preferences.js class BasePreferences { - #browserDefaults = Object.freeze({ - canvasMaxAreaInBytes: -1, - isInAutomation: false, - supportsCaretBrowsingMode: false, - supportsDocumentFonts: true, - supportsIntegratedFind: false, - supportsMouseWheelZoomCtrlKey: true, - supportsMouseWheelZoomMetaKey: true, - supportsPinchToZoom: true - }); #defaults = Object.freeze({ + altTextLearnMoreUrl: "", annotationEditorMode: 0, annotationMode: 2, cursorToolOnLoad: 0, defaultZoomDelay: 400, defaultZoomValue: "", disablePageLabels: false, + enableAltText: false, + enableGuessAltText: true, enableHighlightEditor: false, enableHighlightFloatingButton: false, - enableML: false, enablePermissions: false, enablePrintAutoRotate: true, enableScripting: true, - enableStampEditor: true, + enableUpdatedAddImage: false, externalLinkTarget: 0, highlightEditorColors: "yellow=#FFFF98,green=#53FFBC,blue=#80EBFF,pink=#FFCBE6,red=#FF4F5F", historyUpdateUrl: false, @@ -1456,7 +1504,6 @@ class BasePreferences { enableXfa: true, viewerCssTheme: 0 }); - #prefs = Object.create(null); #initializedPromise = null; constructor() { if (this.constructor === BasePreferences) { @@ -1466,16 +1513,13 @@ class BasePreferences { browserPrefs, prefs }) => { - const options = Object.create(null); - for (const [name, val] of Object.entries(this.#browserDefaults)) { - const prefVal = browserPrefs?.[name]; - options[name] = typeof prefVal === typeof val ? prefVal : val; + if (AppOptions._checkDisablePreferences()) { + return; } - for (const [name, val] of Object.entries(this.#defaults)) { - const prefVal = prefs?.[name]; - options[name] = this.#prefs[name] = typeof prefVal === typeof val ? prefVal : val; - } - AppOptions.setAll(options, true); + AppOptions.setAll({ + ...browserPrefs, + ...prefs + }, true); }); } async _writeToStorage(prefObj) { @@ -1484,58 +1528,21 @@ class BasePreferences { async _readFromStorage(prefObj) { throw new Error("Not implemented: _readFromStorage"); } - #updatePref({ - name, - value - }) { - throw new Error("Not implemented: #updatePref"); - } async reset() { await this.#initializedPromise; - const oldPrefs = structuredClone(this.#prefs); - this.#prefs = Object.create(null); - try { - await this._writeToStorage(this.#defaults); - } catch (reason) { - this.#prefs = oldPrefs; - throw reason; - } + AppOptions.setAll(this.#defaults, true); + await this._writeToStorage(this.#defaults); } async set(name, value) { await this.#initializedPromise; - const defaultValue = this.#defaults[name], - oldPrefs = structuredClone(this.#prefs); - if (defaultValue === undefined) { - throw new Error(`Set preference: "${name}" is undefined.`); - } else if (value === undefined) { - throw new Error("Set preference: no value is specified."); - } - const valueType = typeof value, - defaultType = typeof defaultValue; - if (valueType !== defaultType) { - if (valueType === "number" && defaultType === "string") { - value = value.toString(); - } else { - throw new Error(`Set preference: "${value}" is a ${valueType}, expected a ${defaultType}.`); - } - } else if (valueType === "number" && !Number.isInteger(value)) { - throw new Error(`Set preference: "${value}" must be an integer.`); - } - this.#prefs[name] = value; - try { - await this._writeToStorage(this.#prefs); - } catch (reason) { - this.#prefs = oldPrefs; - throw reason; - } + AppOptions.setAll({ + [name]: value + }, true); + await this._writeToStorage(AppOptions.getAll(OptionKind.PREFERENCE)); } async get(name) { await this.#initializedPromise; - const defaultValue = this.#defaults[name]; - if (defaultValue === undefined) { - throw new Error(`Get preference: "${name}" is undefined.`); - } - return this.#prefs[name] ?? defaultValue; + return AppOptions.get(name); } get initializedPromise() { return this.#initializedPromise; @@ -3098,13 +3105,19 @@ class Preferences extends BasePreferences { } class ExternalServices extends BaseExternalServices { async createL10n() { - return new genericl10n_GenericL10n(AppOptions.get("locale")); + return new genericl10n_GenericL10n(AppOptions.get("localeProperties")?.lang); } createScripting() { return new GenericScripting(AppOptions.get("sandboxBundleSrc")); } } class MLManager { + async isEnabledFor(_name) { + return false; + } + async deleteModel(_service) { + return null; + } async guess() { return null; } @@ -8411,6 +8424,9 @@ class AnnotationLayerBuilder { } this.div.hidden = true; } + hasEditableAnnotations() { + return !!this.annotationLayer?.hasEditableAnnotations(); + } #updatePresentationModeState(state) { if (!this.div) { return; @@ -9142,6 +9158,7 @@ class PDFPageView { #annotationMode = AnnotationMode.ENABLE_FORMS; #enableHWA = false; #hasRestrictedScaling = false; + #isEditing = false; #layerProperties = null; #loadingId = null; #previousRotation = null; @@ -9296,6 +9313,9 @@ class PDFPageView { this.reset(); this.pdfPage?.cleanup(); } + hasEditableAnnotations() { + return !!this.annotationLayer?.hasEditableAnnotations(); + } get _textHighlighter() { return shadow(this, "_textHighlighter", new TextHighlighter({ pageIndex: this.id - 1, @@ -9472,6 +9492,19 @@ class PDFPageView { this._resetZoomLayer(); } } + toggleEditingMode(isEditing) { + if (!this.hasEditableAnnotations()) { + return; + } + this.#isEditing = isEditing; + this.reset({ + keepZoomLayer: true, + keepAnnotationLayer: true, + keepAnnotationEditorLayer: true, + keepXfaLayer: true, + keepTextLayer: true + }); + } update({ scale = 0, rotation = null, @@ -9822,7 +9855,8 @@ class PDFPageView { annotationMode: this.#annotationMode, optionalContentConfigPromise: this._optionalContentConfigPromise, annotationCanvasMap: this._annotationCanvasMap, - pageColors + pageColors, + isEditing: this.#isEditing }; const renderTask = this.renderTask = pdfPage.render(renderContext); renderTask.onContinue = renderContinueCallback; @@ -9982,8 +10016,11 @@ class PDFViewer { #enableHWA = false; #enableHighlightFloatingButton = false; #enablePermissions = false; + #enableUpdatedAddImage = false; #eventAbortController = null; #mlManager = null; + #onPageRenderedCallback = null; + #switchAnnotationEditorModeTimeoutId = null; #getAllTextInProgress = false; #hiddenCopyElement = null; #interruptCopyCondition = false; @@ -9993,7 +10030,7 @@ class PDFViewer { #scaleTimeoutId = null; #textLayerMode = TextLayerMode.ENABLE; constructor(options) { - const viewerVersion = "4.4.168"; + const viewerVersion = "4.5.136"; if (version !== viewerVersion) { throw new Error(`The API version "${version}" does not match the Viewer version "${viewerVersion}".`); } @@ -10020,6 +10057,7 @@ class PDFViewer { this.#annotationEditorMode = options.annotationEditorMode ?? AnnotationEditorType.NONE; this.#annotationEditorHighlightColors = options.annotationEditorHighlightColors || null; this.#enableHighlightFloatingButton = options.enableHighlightFloatingButton === true; + this.#enableUpdatedAddImage = options.enableUpdatedAddImage === true; this.imageResourcesPath = options.imageResourcesPath || ""; this.enablePrintAutoRotate = options.enablePrintAutoRotate || false; this.removePageBorders = options.removePageBorders || false; @@ -10425,7 +10463,7 @@ class PDFViewer { if (pdfDocument.isPureXfa) { console.warn("Warning: XFA-editing is not implemented."); } else if (isValidAnnotationEditorMode(mode)) { - this.#annotationEditorUIManager = new AnnotationEditorUIManager(this.container, viewer, this.#altTextManager, eventBus, pdfDocument, pageColors, this.#annotationEditorHighlightColors, this.#enableHighlightFloatingButton, this.#mlManager); + this.#annotationEditorUIManager = new AnnotationEditorUIManager(this.container, viewer, this.#altTextManager, eventBus, pdfDocument, pageColors, this.#annotationEditorHighlightColors, this.#enableHighlightFloatingButton, this.#enableUpdatedAddImage, this.#mlManager); eventBus.dispatch("annotationeditoruimanager", { source: this, uiManager: this.#annotationEditorUIManager @@ -10584,6 +10622,7 @@ class PDFViewer { this.viewer.removeAttribute("lang"); this.#hiddenCopyElement?.remove(); this.#hiddenCopyElement = null; + this.#cleanupSwitchAnnotationEditorMode(); } #ensurePageViewVisible() { if (this._scrollMode !== ScrollMode.PAGE) { @@ -10956,6 +10995,34 @@ class PDFViewer { location: this._location }); } + #switchToEditAnnotationMode() { + const visible = this._getVisiblePages(); + const pagesToRefresh = []; + const { + ids, + views + } = visible; + for (const page of views) { + const { + view + } = page; + if (!view.hasEditableAnnotations()) { + ids.delete(view.id); + continue; + } + pagesToRefresh.push(page); + } + if (pagesToRefresh.length === 0) { + return null; + } + this.renderingQueue.renderHighestPriority({ + first: pagesToRefresh[0], + last: pagesToRefresh.at(-1), + views: pagesToRefresh, + ids + }); + return ids; + } containsElement(element) { return this.container.contains(element); } @@ -11388,6 +11455,16 @@ class PDFViewer { get containerTopLeft() { return this.#containerTopLeft ||= [this.container.offsetTop, this.container.offsetLeft]; } + #cleanupSwitchAnnotationEditorMode() { + if (this.#onPageRenderedCallback) { + this.eventBus._off("pagerendered", this.#onPageRenderedCallback); + this.#onPageRenderedCallback = null; + } + if (this.#switchAnnotationEditorModeTimeoutId !== null) { + clearTimeout(this.#switchAnnotationEditorModeTimeoutId); + this.#switchAnnotationEditorModeTimeoutId = null; + } + } get annotationEditorMode() { return this.#annotationEditorUIManager ? this.#annotationEditorMode : AnnotationEditorType.DISABLE; } @@ -11408,12 +11485,47 @@ class PDFViewer { if (!this.pdfDocument) { return; } - this.#annotationEditorMode = mode; - this.eventBus.dispatch("annotationeditormodechanged", { - source: this, - mode - }); - this.#annotationEditorUIManager.updateMode(mode, editId, isFromKeyboard); + const { + eventBus + } = this; + const updater = () => { + this.#cleanupSwitchAnnotationEditorMode(); + this.#annotationEditorMode = mode; + this.#annotationEditorUIManager.updateMode(mode, editId, isFromKeyboard); + eventBus.dispatch("annotationeditormodechanged", { + source: this, + mode + }); + }; + if (mode === AnnotationEditorType.NONE || this.#annotationEditorMode === AnnotationEditorType.NONE) { + const isEditing = mode !== AnnotationEditorType.NONE; + if (!isEditing) { + this.pdfDocument.annotationStorage.resetModifiedIds(); + } + for (const pageView of this._pages) { + pageView.toggleEditingMode(isEditing); + } + const idsToRefresh = this.#switchToEditAnnotationMode(); + if (isEditing && idsToRefresh) { + this.#cleanupSwitchAnnotationEditorMode(); + this.#onPageRenderedCallback = ({ + pageNumber + }) => { + idsToRefresh.delete(pageNumber); + if (idsToRefresh.size === 0) { + this.#switchAnnotationEditorModeTimeoutId = setTimeout(updater, 0); + } + }; + const { + signal + } = this.#eventAbortController; + eventBus._on("pagerendered", this.#onPageRenderedCallback, { + signal + }); + return; + } + } + updater(); } set annotationEditorParams({ type, @@ -11721,7 +11833,7 @@ class SecondaryToolbar { class Toolbar { #opts; - constructor(options, eventBus) { + constructor(options, eventBus, toolbarDensity = 0) { this.#opts = options; this.eventBus = eventBus; const buttons = [{ @@ -11806,8 +11918,13 @@ class Toolbar { break; } }); + eventBus._on("toolbardensity", this.#updateToolbarDensity.bind(this)); + this.#updateToolbarDensity({ + value: toolbarDensity + }); this.reset(); } + #updateToolbarDensity() {} #setAnnotationEditorUIManager(uiManager, parentContainer) { const colorPicker = new ColorPicker({ uiManager @@ -12078,7 +12195,6 @@ class ViewHistory { const FORCE_PAGES_LOADED_TIMEOUT = 10000; -const WHEEL_ZOOM_DISABLED_TIMEOUT = 1000; const ViewOnLoad = { UNKNOWN: -1, PREVIOUS: 0, @@ -12110,18 +12226,17 @@ const PDFViewerApplication = { store: null, downloadManager: null, overlayManager: null, - preferences: null, + preferences: new Preferences(), toolbar: null, secondaryToolbar: null, eventBus: null, l10n: null, annotationEditorParams: null, isInitialViewSet: false, - downloadComplete: false, isViewerEmbedded: window.parent !== window, url: "", baseUrl: "", - _allowedGlobalEventsPromise: null, + mlManager: null, _downloadUrl: "", _eventBusAbortController: null, _windowAbortController: null, @@ -12141,11 +12256,9 @@ const PDFViewerApplication = { _printAnnotationStoragePromise: null, _touchInfo: null, _isCtrlKeyDown: false, - _nimbusDataPromise: null, _caretBrowsing: null, _isScrolling: false, async initialize(appConfig) { - let l10nPromise; this.appConfig = appConfig; try { await this.preferences.initializedPromise; @@ -12167,8 +12280,7 @@ const PDFViewerApplication = { if (mode) { document.documentElement.classList.add(mode); } - l10nPromise = this.externalServices.createL10n(); - this.l10n = await l10nPromise; + this.l10n = await this.externalServices.createL10n(); document.getElementsByTagName("html")[0].dir = this.l10n.getDirection(); this.l10n.translate(appConfig.appContainer || document.documentElement); if (this.isViewerEmbedded && AppOptions.get("externalLinkTarget") === LinkTarget.NONE) { @@ -12257,7 +12369,9 @@ const PDFViewerApplication = { } } if (params.has("locale")) { - AppOptions.set("locale", params.get("locale")); + AppOptions.set("localeProperties", { + lang: params.get("locale") + }); } }, async _initializeViewerComponents() { @@ -12318,6 +12432,7 @@ const PDFViewerApplication = { annotationEditorMode, annotationEditorHighlightColors: AppOptions.get("highlightEditorColors"), enableHighlightFloatingButton: AppOptions.get("enableHighlightFloatingButton"), + enableUpdatedAddImage: AppOptions.get("enableUpdatedAddImage"), imageResourcesPath: AppOptions.get("imageResourcesPath"), enablePrintAutoRotate: AppOptions.get("enablePrintAutoRotate"), maxCanvasPixels: AppOptions.get("maxCanvasPixels"), @@ -12355,9 +12470,6 @@ const PDFViewerApplication = { } if (appConfig.annotationEditorParams) { if (annotationEditorMode !== AnnotationEditorType.DISABLE) { - if (AppOptions.get("enableStampEditor")) { - appConfig.toolbar?.editorStampButton?.classList.remove("hidden"); - } const editorHighlightButton = appConfig.toolbar?.editorHighlightButton; if (editorHighlightButton && AppOptions.get("enableHighlightEditor")) { editorHighlightButton.hidden = false; @@ -12380,7 +12492,7 @@ const PDFViewerApplication = { }); } if (appConfig.toolbar) { - this.toolbar = new Toolbar(appConfig.toolbar, eventBus); + this.toolbar = new Toolbar(appConfig.toolbar, eventBus, AppOptions.get("toolbarDensity")); } if (appConfig.secondaryToolbar) { this.secondaryToolbar = new SecondaryToolbar(appConfig.secondaryToolbar, eventBus); @@ -12437,7 +12549,6 @@ const PDFViewerApplication = { } }, async run(config) { - this.preferences = new Preferences(); await this.initialize(config); const { appConfig, @@ -12514,9 +12625,6 @@ const PDFViewerApplication = { get externalServices() { return shadow(this, "externalServices", new ExternalServices()); }, - get mlManager() { - return shadow(this, "mlManager", AppOptions.get("enableML") === true ? new MLManager() : null); - }, get initialized() { return this._initializedCapability.settled; }, @@ -12597,12 +12705,10 @@ const PDFViewerApplication = { let title = pdfjs_getPdfFilenameFromUrl(url, ""); if (!title) { try { - title = decodeURIComponent(getFilenameFromUrl(url)) || url; - } catch { - title = url; - } + title = decodeURIComponent(getFilenameFromUrl(url)); + } catch {} } - this.setTitle(title); + this.setTitle(title || url); }, setTitle(title = this._title) { this._title = title; @@ -12648,7 +12754,6 @@ const PDFViewerApplication = { this.pdfLinkService.externalLinkEnabled = true; this.store = null; this.isInitialViewSet = false; - this.downloadComplete = false; this.url = ""; this.baseUrl = ""; this._downloadUrl = ""; @@ -12724,9 +12829,7 @@ const PDFViewerApplication = { async download(options = {}) { let data; try { - if (this.downloadComplete) { - data = await this.pdfDocument.getData(); - } + data = await this.pdfDocument.getData(); } catch {} this.downloadManager.download(data, this._downloadUrl, this._docFilename, options); }, @@ -12793,11 +12896,8 @@ const PDFViewerApplication = { return message; }, progress(level) { - if (!this.loadingBar || this.downloadComplete) { - return; - } const percent = Math.round(level * 100); - if (percent <= this.loadingBar.percent) { + if (!this.loadingBar || percent <= this.loadingBar.percent) { return; } this.loadingBar.percent = percent; @@ -12811,7 +12911,6 @@ const PDFViewerApplication = { length }) => { this._contentLength = length; - this.downloadComplete = true; this.loadingBar?.hide(); firstPagePromise.then(() => { this.eventBus.dispatch("documentloaded", { @@ -13413,9 +13512,6 @@ const PDFViewerApplication = { }); } addWindowResolutionChange(); - window.addEventListener("visibilitychange", webViewerVisibilityChange, { - signal - }); window.addEventListener("wheel", webViewerWheel, { passive: false, signal @@ -13730,7 +13826,7 @@ function webViewerHashchange(evt) { } } { - /*var webViewerFileInputChange = function (evt) { + var webViewerFileInputChange = function (evt) { if (PDFViewerApplication.pdfViewer?.isInPresentationMode) { return; } @@ -13742,7 +13838,7 @@ function webViewerHashchange(evt) { }; var webViewerOpenFile = function (evt) { PDFViewerApplication._openFileInput?.click(); - };*/ + }; } function webViewerPresentationMode() { PDFViewerApplication.requestPresentationMode(); @@ -13876,20 +13972,6 @@ function webViewerPageChanging({ function webViewerResolutionChange(evt) { PDFViewerApplication.pdfViewer.refresh(); } -function webViewerVisibilityChange(evt) { - if (document.visibilityState === "visible") { - setZoomDisabledTimeout(); - } -} -let zoomDisabledTimeout = null; -function setZoomDisabledTimeout() { - if (zoomDisabledTimeout) { - clearTimeout(zoomDisabledTimeout); - } - zoomDisabledTimeout = setTimeout(function () { - zoomDisabledTimeout = null; - }, WHEEL_ZOOM_DISABLED_TIMEOUT); -} function webViewerWheel(evt) { const { pdfViewer, @@ -13907,7 +13989,7 @@ function webViewerWheel(evt) { const origin = [evt.clientX, evt.clientY]; if (isPinchToZoom || evt.ctrlKey && supportsMouseWheelZoomCtrlKey || evt.metaKey && supportsMouseWheelZoomMetaKey) { evt.preventDefault(); - if (PDFViewerApplication._isScrolling || zoomDisabledTimeout || document.visibilityState === "hidden" || PDFViewerApplication.overlayManager.active) { + if (PDFViewerApplication._isScrolling || document.visibilityState === "hidden" || PDFViewerApplication.overlayManager.active) { return; } if (isPinchToZoom && supportsPinchToZoom) { @@ -14335,14 +14417,20 @@ function webViewerReportTelemetry({ }) { PDFViewerApplication.externalServices.reportTelemetry(details); } +function webViewerSetPreference({ + name, + value +}) { + PDFViewerApplication.preferences.set(name, value); +} ;// CONCATENATED MODULE: ./web/viewer.js -const pdfjsVersion = "4.4.168"; -const pdfjsBuild = "19fbc8998"; +const pdfjsVersion = "4.5.136"; +const pdfjsBuild = "3a21f03b0"; const AppConstants = { LinkTarget: LinkTarget, RenderingStates: RenderingStates, diff --git a/cps/static/js/main.js b/cps/static/js/main.js index 72f01867..505d5d9e 100644 --- a/cps/static/js/main.js +++ b/cps/static/js/main.js @@ -49,7 +49,7 @@ function elementSorter(a, b) { return 0; } -// Generic control/related handler to show/hide fields based on a checkbox' value +// Generic control/related handler to show/hide fields based on a 'checkbox' value // e.g. // //
...
@@ -63,7 +63,7 @@ $(document).on("change", "input[type=\"checkbox\"][data-control]", function () { }); }); -// Generic control/related handler to show/hide fields based on a select' value +// Generic control/related handler to show/hide fields based on a 'select' value $(document).on("change", "select[data-control]", function() { var $this = $(this); var name = $this.data("control"); @@ -79,7 +79,7 @@ $(document).on("change", "select[data-control]", function() { } }); -// Generic control/related handler to show/hide fields based on a select' value +// Generic control/related handler to show/hide fields based on a 'select' value // this one is made to show all values if select value is not 0 $(document).on("change", "select[data-controlall]", function() { var $this = $(this); @@ -130,8 +130,13 @@ $(".container-fluid").bind('drop', function (e) { } }); if (dt.files.length) { - $("#btn-upload")[0].files = dt.files; - $("#form-upload").submit(); + if($("#btn-upload-format").length) { + $("#btn-upload-format")[0].files = dt.files; + $("#form-upload-format").submit(); + } else { + $("#btn-upload")[0].files = dt.files; + $("#form-upload").submit(); + } } } }); @@ -140,12 +145,25 @@ $("#btn-upload").change(function() { $("#form-upload").submit(); }); +$("#btn-upload-format").change(function() { + $("#form-upload-format").submit(); +}); + + $("#form-upload").uploadprogress({ - redirect_url: getPath() + "/", //"{{ url_for('web.index')}}", - uploadedMsg: $("#form-upload").data("message"), //"{{_('Upload done, processing, please wait...')}}", - modalTitle: $("#form-upload").data("title"), //"{{_('Uploading...')}}", - modalFooter: $("#form-upload").data("footer"), //"{{_('Close')}}", - modalTitleFailed: $("#form-upload").data("failed") //"{{_('Error')}}" + redirect_url: getPath() + "/", + uploadedMsg: $("#form-upload").data("message"), + modalTitle: $("#form-upload").data("title"), + modalFooter: $("#form-upload").data("footer"), + modalTitleFailed: $("#form-upload").data("failed") +}); + +$("#form-upload-format").uploadprogress({ + redirect_url: getPath() + "/", + uploadedMsg: $("#form-upload-format").data("message"), + modalTitle: $("#form-upload-format").data("title"), + modalFooter: $("#form-upload-format").data("footer"), + modalTitleFailed: $("#form-upload-format").data("failed") }); $(document).ready(function() { @@ -604,6 +622,7 @@ $(function() { }); $("#toggle_order_shelf").click(function() { + $("#toggle_order_shelf").toggleClass("dummy"); $("#new").toggleClass("disabled"); $("#old").toggleClass("disabled"); $("#asc").toggleClass("disabled"); @@ -612,9 +631,20 @@ $(function() { $("#auth_za").toggleClass("disabled"); $("#pub_new").toggleClass("disabled"); $("#pub_old").toggleClass("disabled"); + $("#shelf_new").toggleClass("disabled"); + $("#shelf_old").toggleClass("disabled"); var alternative_text = $("#toggle_order_shelf").data('alt-text'); + var status = $("#toggle_order_shelf").hasClass("dummy") ? "on" : "off"; $("#toggle_order_shelf").data('alt-text', $("#toggle_order_shelf").html()); $("#toggle_order_shelf").html(alternative_text); + + $.ajax({ + method:"post", + contentType: "application/json; charset=utf-8", + dataType: "json", + url: getPath() + "/ajax/view", + data: "{\"shelf\": {\"man\": \"" + status + "\"}}", + }); }); $("#btndeluser").click(function() { @@ -696,20 +726,20 @@ $(function() { url: getPath() + "/ajax/simulatedbchange", data: {config_calibre_dir: $("#config_calibre_dir").val(), csrf_token: $("input[name='csrf_token']").val()}, success: function success(data) { - if ( data.change ) { - if ( data.valid ) { + if ( !data.valid ) { + $("#InvalidDialog").modal('show'); + } + else{ + if ( data.change ) { confirmDialog( "db_submit", "GeneralChangeModal", 0, changeDbSettings ); - } - else { - $("#InvalidDialog").modal('show'); - } - } else { + } else { changeDbSettings(); + } } } }); diff --git a/cps/static/js/reading/epub-progress.js b/cps/static/js/reading/epub-progress.js deleted file mode 100644 index b8fc11d0..00000000 --- a/cps/static/js/reading/epub-progress.js +++ /dev/null @@ -1,54 +0,0 @@ -/** - * waits until queue is finished, meaning the book is done loading - * @param callback - */ -function qFinished(callback){ - let timeout=setInterval(()=>{ - if(reader.rendition.q.running===undefined) - clearInterval(timeout); - callback(); - },300 - ) -} - -function calculateProgress(){ - let data=reader.rendition.location.end; - return Math.round(epub.locations.percentageFromCfi(data.cfi)*100); -} - -// register new event emitter locationchange that fires on urlchange -// source: https://stackoverflow.com/a/52809105/21941129 -(() => { - let oldPushState = history.pushState; - history.pushState = function pushState() { - let ret = oldPushState.apply(this, arguments); - window.dispatchEvent(new Event('locationchange')); - return ret; - }; - - let oldReplaceState = history.replaceState; - history.replaceState = function replaceState() { - let ret = oldReplaceState.apply(this, arguments); - window.dispatchEvent(new Event('locationchange')); - return ret; - }; - - window.addEventListener('popstate', () => { - window.dispatchEvent(new Event('locationchange')); - }); -})(); - -window.addEventListener('locationchange',()=>{ - let newPos=calculateProgress(); - progressDiv.textContent=newPos+"%"; -}); - -var epub=ePub(calibre.bookUrl) - -let progressDiv=document.getElementById("progress"); - -qFinished(()=>{ - epub.locations.generate().then(()=> { - window.dispatchEvent(new Event('locationchange')) -}); -}) diff --git a/cps/static/js/reading/epub.js b/cps/static/js/reading/epub.js index a552da05..8d299ce8 100644 --- a/cps/static/js/reading/epub.js +++ b/cps/static/js/reading/epub.js @@ -52,6 +52,32 @@ var reader; } }); + // Update progress percentage + let progressDiv = document.getElementById("progress"); + reader.book.ready.then((()=>{ + let locations_key = reader.book.key()+'-locations'; + let stored_locations = localStorage.getItem(locations_key); + let make_locations, save_locations; + if (stored_locations) { + make_locations = Promise.resolve(reader.book.locations.load(stored_locations)); + // No-op because locations are already saved + save_locations = ()=>{}; + } else { + make_locations = reader.book.locations.generate(); + save_locations = ()=>{ + localStorage.setItem(locations_key, reader.book.locations.save()); + }; + } + make_locations.then(()=>{ + reader.rendition.on('relocated', (location)=>{ + let percentage = Math.round(location.end.percentage*100); + progressDiv.textContent=percentage+"%"; + }); + reader.rendition.reportLocation(); + progressDiv.style.visibility = "visible"; + }).then(save_locations); + })); + /** * @param {string} action - Add or remove bookmark * @param {string|int} location - Location or zero diff --git a/cps/static/js/reading/locationchange-polyfill.js b/cps/static/js/reading/locationchange-polyfill.js new file mode 100644 index 00000000..4845ea84 --- /dev/null +++ b/cps/static/js/reading/locationchange-polyfill.js @@ -0,0 +1,21 @@ +// register new event emitter locationchange that fires on urlchange +// source: https://stackoverflow.com/a/52809105/21941129 +(() => { + let oldPushState = history.pushState; + history.pushState = function pushState() { + let ret = oldPushState.apply(this, arguments); + window.dispatchEvent(new Event('locationchange')); + return ret; + }; + + let oldReplaceState = history.replaceState; + history.replaceState = function replaceState() { + let ret = oldReplaceState.apply(this, arguments); + window.dispatchEvent(new Event('locationchange')); + return ret; + }; + + window.addEventListener('popstate', () => { + window.dispatchEvent(new Event('locationchange')); + }); +})(); diff --git a/cps/static/js/table.js b/cps/static/js/table.js index 36361c3c..7f210eb1 100644 --- a/cps/static/js/table.js +++ b/cps/static/js/table.js @@ -685,6 +685,17 @@ function ratingFormatter(value, row) { return (value/2); } +function seriesIndexFormatter(value, row) { + if (!value) { + return value; + } + formated_value = Number(value).toFixed(2); + if (formated_value.endsWith(".00")) { + formated_value = parseInt(formated_value).toString(); + } + return formated_value; +} + /* Do some hiding disabling after user list is loaded */ function loadSuccess() { @@ -849,6 +860,7 @@ function BookCheckboxChange(checkbox, userId, field) { }, success: handleListServerResponse }); + console.log("test"); } diff --git a/setup.py b/cps/string_helper.py similarity index 50% rename from setup.py rename to cps/string_helper.py index 8dcbee25..d263dd2f 100644 --- a/setup.py +++ b/cps/string_helper.py @@ -1,8 +1,7 @@ -#!/usr/bin/env python # -*- coding: utf-8 -*- # This file is part of the Calibre-Web (https://github.com/janeczku/calibre-web) -# Copyright (C) 2019 decentral1se +# Copyright (C) 2024 OzzieIsaacs # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by @@ -16,28 +15,9 @@ # # You should have received a copy of the GNU General Public License # along with this program. If not, see . -# -# """Calibre-web distribution package setuptools installer.""" - -from setuptools import setup -import os import re -import codecs -here = os.path.abspath(os.path.dirname(__file__)) -def read(*parts): - with codecs.open(os.path.join(here, *parts), 'r') as fp: - return fp.read() +def strip_whitespaces(text): + return re.sub(r"(^[\s\u200B-\u200D\ufeff]+)|([\s\u200B-\u200D\ufeff]+$)","", text) -def find_version(*file_paths): - version_file = read(*file_paths) - version_match = re.search(r"^STABLE_VERSION\s+=\s+{['\"]version['\"]:\s*['\"](.*)['\"]}", - version_file, re.M) - if version_match: - return version_match.group(1) - raise RuntimeError("Unable to find version string.") - -setup( - version=find_version("src", "calibreweb", "cps", "constants.py") -) diff --git a/cps/tasks/convert.py b/cps/tasks/convert.py index 3bef81a9..dc0af0c4 100644 --- a/cps/tasks/convert.py +++ b/cps/tasks/convert.py @@ -1,339 +1,355 @@ -# -*- coding: utf-8 -*- - -# This file is part of the Calibre-Web (https://github.com/janeczku/calibre-web) -# Copyright (C) 2020 pwr -# -# This program is free software: you can redistribute it and/or modify -# it under the terms of the GNU General Public License as published by -# the Free Software Foundation, either version 3 of the License, or -# (at your option) any later version. -# -# This program is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -# GNU General Public License for more details. -# -# You should have received a copy of the GNU General Public License -# along with this program. If not, see . - -import os -import re -from glob import glob -from shutil import copyfile, copyfileobj -from markupsafe import escape -from time import time -from uuid import uuid4 - -from sqlalchemy.exc import SQLAlchemyError -from flask_babel import lazy_gettext as N_ - -from cps.services.worker import CalibreTask -from cps import db -from cps import logger, config -from cps.subproc_wrapper import process_open -from flask_babel import gettext as _ -from cps.kobo_sync_status import remove_synced_book -from cps.ub import init_db_thread -from cps.file_helper import get_temp_dir - -from cps.tasks.mail import TaskEmail -from cps import gdriveutils, helper -from cps.constants import SUPPORTED_CALIBRE_BINARIES - -log = logger.create() - -current_milli_time = lambda: int(round(time() * 1000)) - - -class TaskConvert(CalibreTask): - def __init__(self, file_path, book_id, task_message, settings, ereader_mail, user=None): - super(TaskConvert, self).__init__(task_message) - self.worker_thread = None - self.file_path = file_path - self.book_id = book_id - self.title = "" - self.settings = settings - self.ereader_mail = ereader_mail - self.user = user - - self.results = dict() - - def run(self, worker_thread): - self.worker_thread = worker_thread - if config.config_use_google_drive: - worker_db = db.CalibreDB(expire_on_commit=False, init=True) - cur_book = worker_db.get_book(self.book_id) - self.title = cur_book.title - data = worker_db.get_book_format(self.book_id, self.settings['old_book_format']) - df = gdriveutils.getFileFromEbooksFolder(cur_book.path, - data.name + "." + self.settings['old_book_format'].lower()) - df_cover = gdriveutils.getFileFromEbooksFolder(cur_book.path, "cover.jpg") - if df: - datafile_cover = None - datafile = os.path.join(config.get_book_path(), - cur_book.path, - data.name + "." + self.settings['old_book_format'].lower()) - if df_cover: - datafile_cover = os.path.join(config.get_book_path(), - cur_book.path, "cover.jpg") - if not os.path.exists(os.path.join(config.get_book_path(), cur_book.path)): - os.makedirs(os.path.join(config.get_book_path(), cur_book.path)) - df.GetContentFile(datafile) - if df_cover: - df_cover.GetContentFile(datafile_cover) - worker_db.session.close() - else: - # ToDo Include cover in error handling - error_message = _("%(format)s not found on Google Drive: %(fn)s", - format=self.settings['old_book_format'], - fn=data.name + "." + self.settings['old_book_format'].lower()) - worker_db.session.close() - return self._handleError(error_message) - - filename = self._convert_ebook_format() - if config.config_use_google_drive: - os.remove(self.file_path + '.' + self.settings['old_book_format'].lower()) - if df_cover: - os.remove(os.path.join(config.config_calibre_dir, cur_book.path, "cover.jpg")) - - if filename: - if config.config_use_google_drive: - # Upload files to gdrive - gdriveutils.updateGdriveCalibreFromLocal() - self._handleSuccess() - if self.ereader_mail: - # if we're sending to E-Reader after converting, create a one-off task and run it immediately - # todo: figure out how to incorporate this into the progress - try: - EmailText = N_(u"%(book)s send to E-Reader", book=escape(self.title)) - for email in self.ereader_mail.split(','): - email = email.strip() - worker_thread.add(self.user, TaskEmail(self.settings['subject'], - self.results["path"], - filename, - self.settings, - email, - EmailText, - self.settings['body'], - id=self.book_id, - internal=True) - ) - except Exception as ex: - return self._handleError(str(ex)) - - def _convert_ebook_format(self): - error_message = None - local_db = db.CalibreDB(expire_on_commit=False, init=True) - file_path = self.file_path - book_id = self.book_id - format_old_ext = '.' + self.settings['old_book_format'].lower() - format_new_ext = '.' + self.settings['new_book_format'].lower() - - # check to see if destination format already exists - or if book is in database - # if it does - mark the conversion task as complete and return a success - # this will allow to send to E-Reader workflow to continue to work - if os.path.isfile(file_path + format_new_ext) or\ - local_db.get_book_format(self.book_id, self.settings['new_book_format']): - log.info("Book id %d already converted to %s", book_id, format_new_ext) - cur_book = local_db.get_book(book_id) - self.title = cur_book.title - self.results['path'] = cur_book.path - self.results['title'] = self.title - new_format = local_db.session.query(db.Data).filter(db.Data.book == book_id)\ - .filter(db.Data.format == self.settings['new_book_format'].upper()).one_or_none() - if not new_format: - new_format = db.Data(name=os.path.basename(file_path), - book_format=self.settings['new_book_format'].upper(), - book=book_id, uncompressed_size=os.path.getsize(file_path + format_new_ext)) - try: - local_db.session.merge(new_format) - local_db.session.commit() - except SQLAlchemyError as e: - local_db.session.rollback() - log.error("Database error: %s", e) - local_db.session.close() - self._handleError(N_("Oops! Database Error: %(error)s.", error=e)) - return - self._handleSuccess() - local_db.session.close() - return os.path.basename(file_path + format_new_ext) - else: - log.info("Book id %d - target format of %s does not exist. Moving forward with convert.", - book_id, - format_new_ext) - - if config.config_kepubifypath and format_old_ext == '.epub' and format_new_ext == '.kepub': - check, error_message = self._convert_kepubify(file_path, - format_old_ext, - format_new_ext) - else: - # check if calibre converter-executable is existing - if not os.path.exists(config.config_converterpath): - self._handleError(N_("Calibre ebook-convert %(tool)s not found", tool=config.config_converterpath)) - return - has_cover = local_db.get_book(book_id).has_cover - check, error_message = self._convert_calibre(file_path, format_old_ext, format_new_ext, has_cover) - - if check == 0: - cur_book = local_db.get_book(book_id) - if os.path.isfile(file_path + format_new_ext): - new_format = local_db.session.query(db.Data).filter(db.Data.book == book_id) \ - .filter(db.Data.format == self.settings['new_book_format'].upper()).one_or_none() - if not new_format: - new_format = db.Data(name=cur_book.data[0].name, - book_format=self.settings['new_book_format'].upper(), - book=book_id, uncompressed_size=os.path.getsize(file_path + format_new_ext)) - try: - local_db.session.merge(new_format) - local_db.session.commit() - if self.settings['new_book_format'].upper() in ['KEPUB', 'EPUB', 'EPUB3']: - ub_session = init_db_thread() - remove_synced_book(book_id, True, ub_session) - ub_session.close() - except SQLAlchemyError as e: - local_db.session.rollback() - log.error("Database error: %s", e) - local_db.session.close() - self._handleError(error_message) - return - self.results['path'] = cur_book.path - self.title = cur_book.title - self.results['title'] = self.title - if not config.config_use_google_drive: - self._handleSuccess() - return os.path.basename(file_path + format_new_ext) - else: - error_message = N_('%(format)s format not found on disk', format=format_new_ext.upper()) - local_db.session.close() - log.info("ebook converter failed with error while converting book") - if not error_message: - error_message = N_('Ebook converter failed with unknown error') - else: - log.error(error_message) - self._handleError(error_message) - return - - def _convert_kepubify(self, file_path, format_old_ext, format_new_ext): - if config.config_embed_metadata and config.config_binariesdir: - tmp_dir, temp_file_name = helper.do_calibre_export(self.book_id, format_old_ext[1:]) - filename = os.path.join(tmp_dir, temp_file_name + format_old_ext) - temp_file_path = tmp_dir - else: - filename = file_path + format_old_ext - temp_file_path = os.path.dirname(file_path) - quotes = [1, 3] - command = [config.config_kepubifypath, filename, '-o', temp_file_path, '-i'] - try: - p = process_open(command, quotes) - except OSError as e: - return 1, N_("Kepubify-converter failed: %(error)s", error=e) - self.progress = 0.01 - while True: - nextline = p.stdout.readlines() - nextline = [x.strip('\n') for x in nextline if x != '\n'] - for line in nextline: - log.debug(line) - if p.poll() is not None: - break - - # process returncode - check = p.returncode - - # move file - if check == 0: - converted_file = glob(os.path.splitext(filename)[0] + "*.kepub.epub") - if len(converted_file) == 1: - copyfile(converted_file[0], (file_path + format_new_ext)) - os.unlink(converted_file[0]) - else: - return 1, N_("Converted file not found or more than one file in folder %(folder)s", - folder=os.path.dirname(file_path)) - return check, None - - def _convert_calibre(self, file_path, format_old_ext, format_new_ext, has_cover): - path_tmp_opf = None - try: - # path_tmp_opf = self._embed_metadata() - if config.config_embed_metadata: - quotes = [5] - tmp_dir = get_temp_dir() - calibredb_binarypath = os.path.join(config.config_binariesdir, SUPPORTED_CALIBRE_BINARIES["calibredb"]) - my_env = os.environ.copy() - if config.config_calibre_split: - my_env['CALIBRE_OVERRIDE_DATABASE_PATH'] = os.path.join(config.config_calibre_dir, "metadata.db") - library_path = config.config_calibre_split_dir - else: - library_path = config.config_calibre_dir - - opf_command = [calibredb_binarypath, 'show_metadata', '--as-opf', str(self.book_id), - '--with-library', library_path] - p = process_open(opf_command, quotes, my_env) - p.wait() - check = p.returncode - calibre_traceback = p.stderr.readlines() - if check == 0: - path_tmp_opf = os.path.join(tmp_dir, "metadata_" + str(uuid4()) + ".opf") - with open(path_tmp_opf, 'w') as fd: - copyfileobj(p.stdout, fd) - else: - error_message = "" - for ele in calibre_traceback: - if not ele.startswith('Traceback') and not ele.startswith(' File'): - error_message = N_("Calibre failed with error: %(error)s", error=ele) - return check, error_message - quotes = [1, 2, 4, 6] - command = [config.config_converterpath, (file_path + format_old_ext), - (file_path + format_new_ext)] - if config.config_embed_metadata: - command.extend(['--from-opf', path_tmp_opf]) - if has_cover: - command.extend(['--cover', os.path.join(os.path.dirname(file_path), 'cover.jpg')]) - quotes_index = 3 - if config.config_calibre: - parameters = config.config_calibre.split(" ") - for param in parameters: - command.append(param) - quotes.append(quotes_index) - quotes_index += 1 - - p = process_open(command, quotes, newlines=False) - except OSError as e: - return 1, N_("Ebook-converter failed: %(error)s", error=e) - - while p.poll() is None: - nextline = p.stdout.readline() - if isinstance(nextline, bytes): - nextline = nextline.decode('utf-8', errors="ignore").strip('\r\n') - if nextline: - log.debug(nextline) - # parse progress string from calibre-converter - progress = re.search(r"(\d+)%\s.*", nextline) - if progress: - self.progress = int(progress.group(1)) / 100 - if config.config_use_google_drive: - self.progress *= 0.9 - - # process returncode - check = p.returncode - calibre_traceback = p.stderr.readlines() - error_message = "" - for ele in calibre_traceback: - ele = ele.decode('utf-8', errors="ignore").strip('\n') - log.debug(ele) - if not ele.startswith('Traceback') and not ele.startswith(' File'): - error_message = N_("Calibre failed with error: %(error)s", error=ele) - return check, error_message - - @property - def name(self): - return N_("Convert") - - def __str__(self): - if self.ereader_mail: - return "Convert Book {} and mail it to {}".format(self.book_id, self.ereader_mail) - else: - return "Convert Book {}".format(self.book_id) - - @property - def is_cancellable(self): - return False +# -*- coding: utf-8 -*- + +# This file is part of the Calibre-Web (https://github.com/janeczku/calibre-web) +# Copyright (C) 2020 pwr +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . + +import os +import re +import glob +from shutil import copyfile, copyfileobj +from markupsafe import escape +from time import time +from uuid import uuid4 + +from sqlalchemy.exc import SQLAlchemyError +from flask_babel import lazy_gettext as N_ + +from cps.services.worker import CalibreTask +from cps import db, app +from cps import logger, config +from cps.subproc_wrapper import process_open +from flask_babel import gettext as _ +from cps.kobo_sync_status import remove_synced_book +from cps.ub import init_db_thread +from cps.file_helper import get_temp_dir + +from cps.tasks.mail import TaskEmail +from cps import gdriveutils, helper +from cps.constants import SUPPORTED_CALIBRE_BINARIES +from cps.string_helper import strip_whitespaces + +log = logger.create() + +current_milli_time = lambda: int(round(time() * 1000)) + + +class TaskConvert(CalibreTask): + def __init__(self, file_path, book_id, task_message, settings, ereader_mail, user=None): + super(TaskConvert, self).__init__(task_message) + self.worker_thread = None + self.file_path = file_path + self.book_id = book_id + self.title = "" + self.settings = settings + self.ereader_mail = ereader_mail + self.user = user + + self.results = dict() + + def run(self, worker_thread): + df_cover = None + cur_book = None + self.worker_thread = worker_thread + if config.config_use_google_drive: + with app.app_context(): + worker_db = db.CalibreDB(app) + cur_book = worker_db.get_book(self.book_id) + self.title = cur_book.title + data = worker_db.get_book_format(self.book_id, self.settings['old_book_format']) + df = gdriveutils.getFileFromEbooksFolder(cur_book.path, + data.name + "." + self.settings['old_book_format'].lower()) + df_cover = gdriveutils.getFileFromEbooksFolder(cur_book.path, "cover.jpg") + if df: + datafile_cover = None + datafile = os.path.join(config.get_book_path(), + cur_book.path, + data.name + "." + self.settings['old_book_format'].lower()) + if df_cover: + datafile_cover = os.path.join(config.get_book_path(), + cur_book.path, "cover.jpg") + if not os.path.exists(os.path.join(config.get_book_path(), cur_book.path)): + os.makedirs(os.path.join(config.get_book_path(), cur_book.path)) + df.GetContentFile(datafile) + if df_cover: + df_cover.GetContentFile(datafile_cover) + # worker_db.session.close() + else: + # ToDo Include cover in error handling + error_message = _("%(format)s not found on Google Drive: %(fn)s", + format=self.settings['old_book_format'], + fn=data.name + "." + self.settings['old_book_format'].lower()) + # worker_db.session.close() + return self._handleError(error_message) + + filename = self._convert_ebook_format() + if config.config_use_google_drive: + os.remove(self.file_path + '.' + self.settings['old_book_format'].lower()) + if df_cover: + os.remove(os.path.join(config.config_calibre_dir, cur_book.path, "cover.jpg")) + + if filename: + if config.config_use_google_drive: + # Upload files to gdrive + gdriveutils.updateGdriveCalibreFromLocal() + self._handleSuccess() + if self.ereader_mail: + # if we're sending to E-Reader after converting, create a one-off task and run it immediately + # todo: figure out how to incorporate this into the progress + try: + EmailText = N_(u"%(book)s send to E-Reader", book=escape(self.title)) + for email in self.ereader_mail.split(','): + email = strip_whitespaces(email) + worker_thread.add(self.user, TaskEmail(self.settings['subject'], + self.results["path"], + filename, + self.settings, + email, + EmailText, + self.settings['body'], + id=self.book_id, + internal=True) + ) + except Exception as ex: + return self._handleError(str(ex)) + + def _convert_ebook_format(self): + error_message = None + with app.app_context(): + local_db = db.CalibreDB(app) + file_path = self.file_path + book_id = self.book_id + format_old_ext = '.' + self.settings['old_book_format'].lower() + format_new_ext = '.' + self.settings['new_book_format'].lower() + + # check to see if destination format already exists - or if book is in database + # if it does - mark the conversion task as complete and return a success + # this will allow to send to E-Reader workflow to continue to work + if os.path.isfile(file_path + format_new_ext) or\ + local_db.get_book_format(self.book_id, self.settings['new_book_format']): + log.info("Book id %d already converted to %s", book_id, format_new_ext) + cur_book = local_db.get_book(book_id) + self.title = cur_book.title + self.results['path'] = cur_book.path + self.results['title'] = self.title + new_format = local_db.session.query(db.Data).filter(db.Data.book == book_id)\ + .filter(db.Data.format == self.settings['new_book_format'].upper()).one_or_none() + if not new_format: + new_format = db.Data(name=os.path.basename(file_path), + book_format=self.settings['new_book_format'].upper(), + book=book_id, uncompressed_size=os.path.getsize(file_path + format_new_ext)) + try: + local_db.session.merge(new_format) + local_db.session.commit() + except SQLAlchemyError as e: + local_db.session.rollback() + log.error("Database error: %s", e) + local_db.session.close() + self._handleError(N_("Oops! Database Error: %(error)s.", error=e)) + return + self._handleSuccess() + local_db.session.close() + return os.path.basename(file_path + format_new_ext) + else: + log.info("Book id %d - target format of %s does not exist. Moving forward with convert.", + book_id, + format_new_ext) + + if config.config_kepubifypath and format_old_ext == '.epub' and format_new_ext == '.kepub': + check, error_message = self._convert_kepubify(file_path, + format_old_ext, + format_new_ext) + else: + # check if calibre converter-executable is existing + if not os.path.exists(config.config_converterpath): + self._handleError(N_("Calibre ebook-convert %(tool)s not found", tool=config.config_converterpath)) + return + has_cover = local_db.get_book(book_id).has_cover + check, error_message = self._convert_calibre(file_path, format_old_ext, format_new_ext, has_cover) + + if check == 0: + cur_book = local_db.get_book(book_id) + if os.path.isfile(file_path + format_new_ext): + new_format = local_db.session.query(db.Data).filter(db.Data.book == book_id) \ + .filter(db.Data.format == self.settings['new_book_format'].upper()).one_or_none() + if not new_format: + new_format = db.Data(name=cur_book.data[0].name, + book_format=self.settings['new_book_format'].upper(), + book=book_id, uncompressed_size=os.path.getsize(file_path + format_new_ext)) + try: + local_db.session.merge(new_format) + local_db.session.commit() + if self.settings['new_book_format'].upper() in ['KEPUB', 'EPUB', 'EPUB3']: + ub_session = init_db_thread() + remove_synced_book(book_id, True, ub_session) + ub_session.close() + except SQLAlchemyError as e: + local_db.session.rollback() + log.error("Database error: %s", e) + local_db.session.close() + self._handleError(error_message) + return + self.results['path'] = cur_book.path + self.title = cur_book.title + self.results['title'] = self.title + if not config.config_use_google_drive: + self._handleSuccess() + return os.path.basename(file_path + format_new_ext) + else: + error_message = N_('%(format)s format not found on disk', format=format_new_ext.upper()) + local_db.session.close() + log.info("ebook converter failed with error while converting book") + if not error_message: + error_message = N_('Ebook converter failed with unknown error') + else: + log.error(error_message) + self._handleError(error_message) + return + + def _convert_kepubify(self, file_path, format_old_ext, format_new_ext): + if config.config_embed_metadata and config.config_binariesdir: + tmp_dir, temp_file_name = helper.do_calibre_export(self.book_id, format_old_ext[1:]) + filename = os.path.join(tmp_dir, temp_file_name + format_old_ext) + temp_file_path = tmp_dir + else: + filename = file_path + format_old_ext + temp_file_path = os.path.dirname(file_path) + quotes = [1, 3] + command = [config.config_kepubifypath, filename, '-o', temp_file_path, '-i'] + try: + p = process_open(command, quotes) + except OSError as e: + return 1, N_("Kepubify-converter failed: %(error)s", error=e) + self.progress = 0.01 + while True: + nextline = p.stdout.readlines() + nextline = [x.strip('\n') for x in nextline if x != '\n'] + for line in nextline: + log.debug(line) + if p.poll() is not None: + break + + # process returncode + check = p.returncode + + # move file + if check == 0: + converted_file = glob.glob(glob.escape(os.path.splitext(filename)[0]) + "*.kepub.epub") + if len(converted_file) == 1: + copyfile(converted_file[0], (file_path + format_new_ext)) + os.unlink(converted_file[0]) + else: + return 1, N_("Converted file not found or more than one file in folder %(folder)s", + folder=os.path.dirname(file_path)) + return check, None + + def _convert_calibre(self, file_path, format_old_ext, format_new_ext, has_cover): + path_tmp_opf = None + try: + # path_tmp_opf = self._embed_metadata() + if config.config_embed_metadata: + quotes = [5] + tmp_dir = get_temp_dir() + calibredb_binarypath = os.path.join(config.config_binariesdir, SUPPORTED_CALIBRE_BINARIES["calibredb"]) + my_env = os.environ.copy() + if config.config_calibre_split: + my_env['CALIBRE_OVERRIDE_DATABASE_PATH'] = os.path.join(config.config_calibre_dir, "metadata.db") + library_path = config.config_calibre_split_dir + else: + library_path = config.config_calibre_dir + + opf_command = [calibredb_binarypath, 'show_metadata', '--as-opf', str(self.book_id), + '--with-library', library_path] + p = process_open(opf_command, quotes, my_env, newlines=False) + lines = list() + while p.poll() is None: + lines.append(p.stdout.readline()) + check = p.returncode + calibre_traceback = p.stderr.readlines() + if check == 0: + path_tmp_opf = os.path.join(tmp_dir, "metadata_" + str(uuid4()) + ".opf") + with open(path_tmp_opf, 'wb') as fd: + fd.write(b''.join(lines)) + else: + error_message = "" + for ele in calibre_traceback: + if not ele.startswith('Traceback') and not ele.startswith(' File'): + error_message = N_("Calibre failed with error: %(error)s", error=ele) + return check, error_message + quotes = [1, 2] + quotes_index = 3 + command = [config.config_converterpath, (file_path + format_old_ext), + (file_path + format_new_ext)] + if config.config_embed_metadata: + quotes.append(4) + quotes_index = 5 + command.extend(['--from-opf', path_tmp_opf]) + if has_cover: + quotes.append(6) + command.extend(['--cover', os.path.join(os.path.dirname(file_path), 'cover.jpg')]) + quotes_index = 7 + if config.config_calibre: + parameters = re.findall(r"(--[\w-]+)(?:(\s(?:(\".+\")|(?:.+?)))(?:\s|$))?", + config.config_calibre, re.IGNORECASE | re.UNICODE) + if parameters: + for param in parameters: + command.append(strip_whitespaces(param[0])) + quotes_index += 1 + if param[1] != "": + parsed = strip_whitespaces(param[1]).strip("\"") + command.append(parsed) + quotes.append(quotes_index) + quotes_index += 1 + p = process_open(command, quotes, newlines=False) + except OSError as e: + return 1, N_("Ebook-converter failed: %(error)s", error=e) + + while p.poll() is None: + nextline = p.stdout.readline() + if isinstance(nextline, bytes): + nextline = nextline.decode('utf-8', errors="ignore").strip('\r\n') + if nextline: + log.debug(nextline) + # parse progress string from calibre-converter + progress = re.search(r"(\d+)%\s.*", nextline) + if progress: + self.progress = int(progress.group(1)) / 100 + if config.config_use_google_drive: + self.progress *= 0.9 + + # process returncode + check = p.returncode + calibre_traceback = p.stderr.readlines() + error_message = "" + for ele in calibre_traceback: + ele = ele.decode('utf-8', errors="ignore").strip('\n') + log.debug(ele) + if not ele.startswith('Traceback') and not ele.startswith(' File'): + error_message = N_("Calibre failed with error: %(error)s", error=ele) + return check, error_message + + @property + def name(self): + return N_("Convert") + + def __str__(self): + if self.ereader_mail: + return "Convert Book {} and mail it to {}".format(self.book_id, self.ereader_mail) + else: + return "Convert Book {}".format(self.book_id) + + @property + def is_cancellable(self): + return False diff --git a/cps/tasks/database.py b/cps/tasks/database.py index c9c30d43..e5f630c6 100644 --- a/cps/tasks/database.py +++ b/cps/tasks/database.py @@ -18,7 +18,7 @@ from flask_babel import lazy_gettext as N_ -from cps import config, logger, db, ub +from cps import config, logger, db, ub, app from cps.services.worker import CalibreTask @@ -26,11 +26,13 @@ class TaskReconnectDatabase(CalibreTask): def __init__(self, task_message=N_('Reconnecting Calibre database')): super(TaskReconnectDatabase, self).__init__(task_message) self.log = logger.create() - self.calibre_db = db.CalibreDB(expire_on_commit=False, init=True) + # self.calibre_db = db.CalibreDB(expire_on_commit=False, init=True) def run(self, worker_thread): - self.calibre_db.reconnect_db(config, ub.app_DB_path) - self.calibre_db.session.close() + with app.app_context(): + calibre_db = db.CalibreDB(app) + calibre_db.reconnect_db(config, ub.app_DB_path) + # self.calibre_db.session.close() self._handleSuccess() @property diff --git a/cps/tasks/mail.py b/cps/tasks/mail.py index 4f85eefa..707a29a3 100644 --- a/cps/tasks/mail.py +++ b/cps/tasks/mail.py @@ -25,7 +25,7 @@ import mimetypes from io import StringIO from email.message import EmailMessage -from email.utils import formatdate, parseaddr +from email.utils import formatdate, parseaddr, make_msgid from email.generator import Generator from flask_babel import lazy_gettext as N_ @@ -34,7 +34,8 @@ from cps.services import gmail from cps.embed_helper import do_calibre_export from cps import logger, config from cps import gdriveutils -import uuid +from cps.string_helper import strip_whitespaces + log = logger.create() @@ -54,8 +55,8 @@ class EmailBase: return (code, resp) def send(self, strg): - """Send `strg' to the server.""" - log.debug_no_auth('send: {}'.format(strg[:300])) + """Send 'strg' to the server.""" + log.debug_no_auth('send: {}'.format(strg[:300]), stacklevel=2) if hasattr(self, 'sock') and self.sock: try: if self.transferSize: @@ -101,7 +102,7 @@ class Email(EmailBase, smtplib.SMTP): smtplib.SMTP.__init__(self, *args, **kwargs) -# Class for sending ssl encrypted email with ability to get current progress, , derived from emailbase class +# Class for sending ssl encrypted email with ability to get current progress, derived from emailbase class class EmailSSL(EmailBase, smtplib.SMTP_SSL): def __init__(self, *args, **kwargs): @@ -127,9 +128,9 @@ class TaskEmail(CalibreTask): try: # Parse out the address from the From line, and then the domain from that from_email = parseaddr(self.settings["mail_from"])[1] - msgid_domain = from_email.partition('@')[2].strip() + msgid_domain = strip_whitespaces(from_email.partition('@')[2]) # This can sometimes sneak through parseaddr if the input is malformed - msgid_domain = msgid_domain.rstrip('>').strip() + msgid_domain = strip_whitespaces(msgid_domain.rstrip('>')) except Exception: msgid_domain = '' return msgid_domain or 'calibre-web.com' @@ -141,7 +142,7 @@ class TaskEmail(CalibreTask): message['To'] = self.recipient message['Subject'] = self.subject message['Date'] = formatdate(localtime=True) - message['Message-Id'] = "{}@{}".format(uuid.uuid4(), self.get_msgid_domain()) + message['Message-ID'] = make_msgid(domain=self.get_msgid_domain()) message.set_content(self.text.encode('UTF-8'), "text", "plain") if self.attachment: data = self._get_attachment(self.filepath, self.attachment) @@ -168,10 +169,14 @@ class TaskEmail(CalibreTask): else: self.send_gmail_email(msg) except MemoryError as e: - log.error_or_exception(e, stacklevel=3) + log.error_or_exception(e, stacklevel=2) self._handleError('MemoryError sending e-mail: {}'.format(str(e))) + except (smtplib.SMTPRecipientsRefused) as e: + log.error_or_exception(e, stacklevel=2) + self._handleError('Smtplib Error sending e-mail: {}'.format( + (list(e.args[0].values())[0][1]).decode('utf-8)').replace("\n", '. '))) except (smtplib.SMTPException, smtplib.SMTPAuthenticationError) as e: - log.error_or_exception(e, stacklevel=3) + log.error_or_exception(e, stacklevel=2) if hasattr(e, "smtp_error"): text = e.smtp_error.decode('utf-8').replace("\n", '. ') elif hasattr(e, "message"): @@ -182,10 +187,10 @@ class TaskEmail(CalibreTask): text = '' self._handleError('Smtplib Error sending e-mail: {}'.format(text)) except (socket.error) as e: - log.error_or_exception(e, stacklevel=3) + log.error_or_exception(e, stacklevel=2) self._handleError('Socket Error sending e-mail: {}'.format(e.strerror)) except Exception as ex: - log.error_or_exception(ex, stacklevel=3) + log.error_or_exception(ex, stacklevel=2) self._handleError('Error sending e-mail: {}'.format(ex)) def send_standard_email(self, msg): @@ -268,7 +273,7 @@ class TaskEmail(CalibreTask): if config.config_binariesdir and config.config_embed_metadata: os.remove(datafile) except IOError as e: - log.error_or_exception(e, stacklevel=3) + log.error_or_exception(e, stacklevel=2) log.error('The requested file could not be read. Maybe wrong permissions?') return None return data diff --git a/cps/tasks/metadata_backup.py b/cps/tasks/metadata_backup.py index 5e0bb96a..048fcf7c 100644 --- a/cps/tasks/metadata_backup.py +++ b/cps/tasks/metadata_backup.py @@ -19,7 +19,7 @@ import os from lxml import etree -from cps import config, db, gdriveutils, logger +from cps import config, db, gdriveutils, logger, app from cps.services.worker import CalibreTask from flask_babel import lazy_gettext as N_ @@ -34,7 +34,7 @@ class TaskBackupMetadata(CalibreTask): task_message=N_('Backing up Metadata')): super(TaskBackupMetadata, self).__init__(task_message) self.log = logger.create() - self.calibre_db = db.CalibreDB(expire_on_commit=False, init=True) + # self.calibre_db = db.CalibreDB(expire_on_commit=False, init=True) self.export_language = export_language self.translated_title = translated_title self.set_dirty = set_dirty @@ -46,47 +46,51 @@ class TaskBackupMetadata(CalibreTask): self.backup_metadata() def set_all_books_dirty(self): - try: - books = self.calibre_db.session.query(db.Books).all() - for book in books: - self.calibre_db.set_metadata_dirty(book.id) - self.calibre_db.session.commit() - self._handleSuccess() - except Exception as ex: - self.log.debug('Error adding book for backup: ' + str(ex)) - self._handleError('Error adding book for backup: ' + str(ex)) - self.calibre_db.session.rollback() - self.calibre_db.session.close() + with app.app_context(): + calibre_dbb = db.CalibreDB(app) + try: + books = calibre_dbb.session.query(db.Books).all() + for book in books: + calibre_dbb.set_metadata_dirty(book.id) + calibre_dbb.session.commit() + self._handleSuccess() + except Exception as ex: + self.log.debug('Error adding book for backup: ' + str(ex)) + self._handleError('Error adding book for backup: ' + str(ex)) + calibre_dbb.session.rollback() + # self.calibre_db.session.close() def backup_metadata(self): - try: - metadata_backup = self.calibre_db.session.query(db.Metadata_Dirtied).all() - custom_columns = (self.calibre_db.session.query(db.CustomColumns) - .filter(db.CustomColumns.mark_for_delete == 0) - .filter(db.CustomColumns.datatype.notin_(db.cc_exceptions)) - .order_by(db.CustomColumns.label).all()) - count = len(metadata_backup) - i = 0 - for backup in metadata_backup: - book = self.calibre_db.session.query(db.Books).filter(db.Books.id == backup.book).one_or_none() - self.calibre_db.session.query(db.Metadata_Dirtied).filter( - db.Metadata_Dirtied.book == backup.book).delete() - self.calibre_db.session.commit() - if book: - self.open_metadata(book, custom_columns) - else: - self.log.error("Book {} not found in database".format(backup.book)) - i += 1 - self.progress = (1.0 / count) * i - self._handleSuccess() - self.calibre_db.session.close() + with app.app_context(): + try: + calibre_dbb = db.CalibreDB(app) + metadata_backup = calibre_dbb.session.query(db.Metadata_Dirtied).all() + custom_columns = (calibre_dbb.session.query(db.CustomColumns) + .filter(db.CustomColumns.mark_for_delete == 0) + .filter(db.CustomColumns.datatype.notin_(db.cc_exceptions)) + .order_by(db.CustomColumns.label).all()) + count = len(metadata_backup) + i = 0 + for backup in metadata_backup: + book = calibre_dbb.session.query(db.Books).filter(db.Books.id == backup.book).one_or_none() + calibre_dbb.session.query(db.Metadata_Dirtied).filter( + db.Metadata_Dirtied.book == backup.book).delete() + calibre_dbb.session.commit() + if book: + self.open_metadata(book, custom_columns) + else: + self.log.error("Book {} not found in database".format(backup.book)) + i += 1 + self.progress = (1.0 / count) * i + self._handleSuccess() + # self.calibre_db.session.close() - except Exception as ex: - b = "NaN" if not hasattr(book, 'id') else book.id - self.log.debug('Error creating metadata backup for book {}: '.format(b) + str(ex)) - self._handleError('Error creating metadata backup: ' + str(ex)) - self.calibre_db.session.rollback() - self.calibre_db.session.close() + except Exception as ex: + b = "NaN" if not hasattr(book, 'id') else book.id + self.log.debug('Error creating metadata backup for book {}: '.format(b) + str(ex)) + self._handleError('Error creating metadata backup: ' + str(ex)) + calibre_dbb.session.rollback() + # self.calibre_db.session.close() def open_metadata(self, book, custom_columns): # package = self.create_new_metadata_backup(book, custom_columns) diff --git a/cps/tasks/thumbnail.py b/cps/tasks/thumbnail.py index d2179dd9..358f4ad1 100644 --- a/cps/tasks/thumbnail.py +++ b/cps/tasks/thumbnail.py @@ -20,11 +20,11 @@ import os from shutil import copyfile, copyfileobj from urllib.request import urlopen from io import BytesIO +from datetime import datetime, timezone from .. import constants -from cps import config, db, fs, gdriveutils, logger, ub +from cps import config, db, fs, gdriveutils, logger, ub, app from cps.services.worker import CalibreTask, STAT_CANCELLED, STAT_ENDED -from datetime import datetime from sqlalchemy import func, text, or_ from flask_babel import lazy_gettext as N_ @@ -36,7 +36,7 @@ except (ImportError, RuntimeError) as e: def get_resize_height(resolution): - return int(225 * resolution) + return int(255 * resolution) def get_resize_width(resolution, original_width, original_height): @@ -73,7 +73,8 @@ class TaskGenerateCoverThumbnails(CalibreTask): self.cache = fs.FileSystem() self.resolutions = [ constants.COVER_THUMBNAIL_SMALL, - constants.COVER_THUMBNAIL_MEDIUM + constants.COVER_THUMBNAIL_MEDIUM, + constants.COVER_THUMBNAIL_LARGE ] def run(self, worker_thread): @@ -113,9 +114,10 @@ class TaskGenerateCoverThumbnails(CalibreTask): @staticmethod def get_books_with_covers(book_id=-1): filter_exp = (db.Books.id == book_id) if book_id != -1 else True - calibre_db = db.CalibreDB(expire_on_commit=False, init=True) - books_cover = calibre_db.session.query(db.Books).filter(db.Books.has_cover == 1).filter(filter_exp).all() - calibre_db.session.close() + with app.app_context(): + calibre_db = db.CalibreDB(app) #, expire_on_commit=False, init=True) + books_cover = calibre_db.session.query(db.Books).filter(db.Books.has_cover == 1).filter(filter_exp).all() + # calibre_db.session.close() return books_cover def get_book_cover_thumbnails(self, book_id): @@ -123,7 +125,7 @@ class TaskGenerateCoverThumbnails(CalibreTask): .query(ub.Thumbnail) \ .filter(ub.Thumbnail.type == constants.THUMBNAIL_TYPE_COVER) \ .filter(ub.Thumbnail.entity_id == book_id) \ - .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.utcnow())) \ + .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.now(timezone.utc))) \ .all() def create_book_cover_thumbnails(self, book): @@ -165,7 +167,7 @@ class TaskGenerateCoverThumbnails(CalibreTask): self.app_db_session.rollback() def update_book_cover_thumbnail(self, book, thumbnail): - thumbnail.generated_at = datetime.utcnow() + thumbnail.generated_at = datetime.now(timezone.utc) try: self.app_db_session.commit() @@ -197,9 +199,11 @@ class TaskGenerateCoverThumbnails(CalibreTask): img.format = thumbnail.format img.save(filename=filename) else: - with open(filename, 'rb') as fd: + stream.seek(0) + with open(filename, 'wb') as fd: copyfileobj(stream, fd) + except Exception as ex: # Bubble exception to calling function self.log.debug('Error generating thumbnail file: ' + str(ex)) @@ -244,7 +248,7 @@ class TaskGenerateSeriesThumbnails(CalibreTask): super(TaskGenerateSeriesThumbnails, self).__init__(task_message) self.log = logger.create() self.app_db_session = ub.get_new_session_instance() - self.calibre_db = db.CalibreDB(expire_on_commit=False, init=True) + # self.calibre_db = db.CalibreDB(expire_on_commit=False, init=True) self.cache = fs.FileSystem() self.resolutions = [ constants.COVER_THUMBNAIL_SMALL, @@ -252,58 +256,60 @@ class TaskGenerateSeriesThumbnails(CalibreTask): ] def run(self, worker_thread): - if self.calibre_db.session and use_IM and self.stat != STAT_CANCELLED and self.stat != STAT_ENDED: - self.message = 'Scanning Series' - all_series = self.get_series_with_four_plus_books() - count = len(all_series) + with app.app_context(): + calibre_db = db.CalibreDB(app) + if calibre_db.session and use_IM and self.stat != STAT_CANCELLED and self.stat != STAT_ENDED: + self.message = 'Scanning Series' + all_series = self.get_series_with_four_plus_books(calibre_db) + count = len(all_series) - total_generated = 0 - for i, series in enumerate(all_series): - generated = 0 - series_thumbnails = self.get_series_thumbnails(series.id) - series_books = self.get_series_books(series.id) + total_generated = 0 + for i, series in enumerate(all_series): + generated = 0 + series_thumbnails = self.get_series_thumbnails(series.id) + series_books = self.get_series_books(series.id, calibre_db) - # Generate new thumbnails for missing covers - resolutions = list(map(lambda t: t.resolution, series_thumbnails)) - missing_resolutions = list(set(self.resolutions).difference(resolutions)) - for resolution in missing_resolutions: - generated += 1 - self.create_series_thumbnail(series, series_books, resolution) - - # Replace outdated or missing thumbnails - for thumbnail in series_thumbnails: - if any(book.last_modified > thumbnail.generated_at for book in series_books): + # Generate new thumbnails for missing covers + resolutions = list(map(lambda t: t.resolution, series_thumbnails)) + missing_resolutions = list(set(self.resolutions).difference(resolutions)) + for resolution in missing_resolutions: generated += 1 - self.update_series_thumbnail(series_books, thumbnail) + self.create_series_thumbnail(series, series_books, resolution) - elif not self.cache.get_cache_file_exists(thumbnail.filename, constants.CACHE_TYPE_THUMBNAILS): - generated += 1 - self.update_series_thumbnail(series_books, thumbnail) + # Replace outdated or missing thumbnails + for thumbnail in series_thumbnails: + if any(book.last_modified > thumbnail.generated_at for book in series_books): + generated += 1 + self.update_series_thumbnail(series_books, thumbnail) - # Increment the progress - self.progress = (1.0 / count) * i + elif not self.cache.get_cache_file_exists(thumbnail.filename, constants.CACHE_TYPE_THUMBNAILS): + generated += 1 + self.update_series_thumbnail(series_books, thumbnail) - if generated > 0: - total_generated += generated - self.message = N_('Generated {0} series thumbnails').format(total_generated) + # Increment the progress + self.progress = (1.0 / count) * i - # Check if job has been cancelled or ended - if self.stat == STAT_CANCELLED: - self.log.info(f'GenerateSeriesThumbnails task has been cancelled.') - return + if generated > 0: + total_generated += generated + self.message = N_('Generated {0} series thumbnails').format(total_generated) - if self.stat == STAT_ENDED: - self.log.info(f'GenerateSeriesThumbnails task has been ended.') - return + # Check if job has been cancelled or ended + if self.stat == STAT_CANCELLED: + self.log.info(f'GenerateSeriesThumbnails task has been cancelled.') + return - if total_generated == 0: - self.self_cleanup = True + if self.stat == STAT_ENDED: + self.log.info(f'GenerateSeriesThumbnails task has been ended.') + return - self._handleSuccess() - self.app_db_session.remove() + if total_generated == 0: + self.self_cleanup = True - def get_series_with_four_plus_books(self): - return self.calibre_db.session \ + self._handleSuccess() + self.app_db_session.remove() + + def get_series_with_four_plus_books(self, calibre_db): + return calibre_db.session \ .query(db.Series) \ .join(db.books_series_link) \ .join(db.Books) \ @@ -312,8 +318,8 @@ class TaskGenerateSeriesThumbnails(CalibreTask): .having(func.count('book_series_link') > 3) \ .all() - def get_series_books(self, series_id): - return self.calibre_db.session \ + def get_series_books(self, series_id, calibre_db): + return calibre_db.session \ .query(db.Books) \ .join(db.books_series_link) \ .join(db.Series) \ @@ -322,12 +328,12 @@ class TaskGenerateSeriesThumbnails(CalibreTask): .all() def get_series_thumbnails(self, series_id): - return self.app_db_session \ - .query(ub.Thumbnail) \ - .filter(ub.Thumbnail.type == constants.THUMBNAIL_TYPE_SERIES) \ - .filter(ub.Thumbnail.entity_id == series_id) \ - .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.utcnow())) \ - .all() + return (self.app_db_session + .query(ub.Thumbnail) + .filter(ub.Thumbnail.type == constants.THUMBNAIL_TYPE_SERIES) + .filter(ub.Thumbnail.entity_id == series_id) + .filter(or_(ub.Thumbnail.expiration.is_(None), ub.Thumbnail.expiration > datetime.now(timezone.utc))) + .all()) def create_series_thumbnail(self, series, series_books, resolution): thumbnail = ub.Thumbnail() @@ -346,7 +352,7 @@ class TaskGenerateSeriesThumbnails(CalibreTask): self.app_db_session.rollback() def update_series_thumbnail(self, series_books, thumbnail): - thumbnail.generated_at = datetime.utcnow() + thumbnail.generated_at = datetime.now(timezone.utc) try: self.app_db_session.commit() @@ -459,13 +465,15 @@ class TaskClearCoverThumbnailCache(CalibreTask): def run(self, worker_thread): if self.app_db_session: - if self.book_id == 0: # delete superfluous thumbnails - calibre_db = db.CalibreDB(expire_on_commit=False, init=True) - thumbnails = (calibre_db.session.query(ub.Thumbnail) - .join(db.Books, ub.Thumbnail.entity_id == db.Books.id, isouter=True) - .filter(db.Books.id==None) - .all()) - calibre_db.session.close() + # delete superfluous thumbnails + if self.book_id == 0: + with app.app_context(): + calibre_db = db.CalibreDB(app) + thumbnails = (calibre_db.session.query(ub.Thumbnail) + .join(db.Books, ub.Thumbnail.entity_id == db.Books.id, isouter=True) + .filter(db.Books.id==None) + .all()) + # calibre_db.session.close() elif self.book_id > 0: # make sure single book is selected thumbnails = self.get_thumbnails_for_book(self.book_id) if self.book_id < 0: diff --git a/cps/templates/author.html b/cps/templates/author.html index f7314586..a01bb40b 100644 --- a/cps/templates/author.html +++ b/cps/templates/author.html @@ -62,18 +62,16 @@ {{author.name.replace('|',',')|shortentitle(30)}} {% endif %} {% endfor %} - {% for format in entry.Books.data %} - {% if format.format|lower in g.constants.EXTENSIONS_AUDIO %} + {% if entry.Books.data|music %} - {% endif %} - {% endfor %} + {% endif %}

{% if entry.Books.series.__len__() > 0 %}

{{entry.Books.series[0].name}} - ({{entry.Books.series_index|formatseriesindex}}) + ({{entry.Books.series_index|formatfloat(2)}})

{% endif %} {% if entry.Books.ratings.__len__() > 0 %} @@ -124,7 +122,7 @@ {{entry.series[0].name}} - ({{entry.series_index|formatseriesindex}}) + ({{entry.series_index|formatfloat(2)}})

{% endif %}
diff --git a/cps/templates/basic_detail.html b/cps/templates/basic_detail.html new file mode 100644 index 00000000..1996497b --- /dev/null +++ b/cps/templates/basic_detail.html @@ -0,0 +1,81 @@ +{% extends "basic_layout.html" %} + +{% block body %} +
+ +

{{ entry.title }}

+
+ {% for author in entry.ordered_authors %} +

{{ author.name.replace("|",",") }}

+ {% endfor %} +
+ +
+ +
+ +{% if current_user.role_download() %} + {% if entry.data|length %} +
+

Download

+ {% for format in entry.data %} +

+ + {{ format.format }} ({{ format.uncompressed_size|filesizeformat }}) +

+ {% endfor %} +
+ {% endif %} +{% endif %} + +

Details

+ +{% if entry.series|length > 0 %} +

{{ _("Book %(index)s of %(range)s", index=entry.series_index | formatfloat(2), range=(entry.series[0].name)|safe) }}

+{% endif %} + +{% if entry.languages|length > 0 %} +
+

+ + {{_('Language')}}: {% for language in entry.languages %}{{language.language_name}}{% if not loop.last %}, {% endif %}{% endfor %} + +

+
+{% endif %} + +{% if entry.identifiers|length > 0 %} +
+

+ + {% for identifier in entry.identifiers %} +

{{ identifier.format_type() }}: {{ identifier|escape }}

+ {% endfor %} +

+
+{% endif %} + +{% if entry.publishers|length > 0 %} +
+

+ {{ _('Publisher') }}: + {{ entry.publishers[0].name }} + +

+
+{% endif %} + +{% if (entry.pubdate|string)[:10] != '0101-01-01' %} +
+

{{ _('Published') }}: {{ entry.pubdate|formatdate }}

+
+{% endif %} + +{% if entry.comments|length > 0 and entry.comments[0].text|length > 0 %} +
+

{{ _('Description:') }}

+ {{ entry.comments[0].text|safe }} +
+{% endif %} +
+{% endblock %} diff --git a/cps/templates/basic_index.html b/cps/templates/basic_index.html new file mode 100644 index 00000000..ae88087f --- /dev/null +++ b/cps/templates/basic_index.html @@ -0,0 +1,32 @@ +{% extends "basic_layout.html" %} +{% block body %} + + + +{% if entries|length < 1 %} +

{{_('No Results Found')}}

+{% endif %} + +{% for entry in entries %} + {% if entry.Books.authors %} + {% set author = entry.Books.authors[0].name.replace('|',',')|shortentitle(30) %} + {% else %} + {% set author = '' %} + {% endif %} + +

{{ author }} - {{entry.Books.title|shortentitle}}

+
+{% endfor %} + +{% endblock %} diff --git a/cps/templates/basic_layout.html b/cps/templates/basic_layout.html new file mode 100644 index 00000000..b3000be8 --- /dev/null +++ b/cps/templates/basic_layout.html @@ -0,0 +1,47 @@ + + + + +{{instance}} | {{title}} + + + + + + +
+
+ {% if current_user.is_authenticated or g.allow_anonymous %} + + + {% endif %} +
+
+
+ {% block body %} + {% endblock %} +
+ + + diff --git a/cps/templates/book_edit.html b/cps/templates/book_edit.html index 57fe701c..615b40c2 100644 --- a/cps/templates/book_edit.html +++ b/cps/templates/book_edit.html @@ -47,42 +47,37 @@
{% endif %} +{% if current_user.role_upload() and g.allow_upload %} +
+
+
+ + +
+ +
+ +
+
+
+{% endif %} +
- - + +
- +
- -
- - -
- -
- - - {% for identifier in book.identifiers %} - - - - - - {% endfor %} -
{{_('Remove')}}
- {{_('Add Identifier')}} -
-
@@ -93,23 +88,8 @@
- +
-
- - -
- {% if current_user.role_upload() and g.allow_upload %} -
- - -
-
- -
- -
- {% endif %}
@@ -126,6 +106,39 @@
+
+ + +
+
+ + +
+
+ + + {% for identifier in book.identifiers %} + + + + + + {% endfor %} + +
{{_('Remove')}}
+ {{_('Add Identifier')}} +
+ {% if current_user.role_upload() and g.allow_upload %} +
+ + +
+
+ +
+ +
+ {% endif %} {% if cc|length > 0 %} {% for c in cc %}
@@ -196,13 +209,6 @@
{% endfor %} {% endif %} - {% if current_user.role_upload() and g.allow_upload %} -
- -
- -
- {% endif %}
- +
diff --git a/cps/templates/detail.html b/cps/templates/detail.html index bca7b1c8..b273ef9e 100644 --- a/cps/templates/detail.html +++ b/cps/templates/detail.html @@ -1,4 +1,12 @@ {% extends is_xhr|yesno("fragment.html", "layout.html") %} +{% block header %} + + + {% if entry.comments|length > 0 and entry.comments[0].text|length > 0 %} + + + {% endif %} +{% endblock %} {% block body %}
{% endif %} {% if entry.series|length > 0 %} -

{{ _("Book %(index)s of %(range)s", index=entry.series_index | formatfloat(2), range=(url_for('web.books_list', data='series', sort_param='stored', book_id=entry.series[0].id)|escapedlink(entry.series[0].name))|safe) }}

+

{{ _("Book %(index)s of %(range)s", index=entry.series_index|formatfloat(2), range=(url_for('web.books_list', data='series', sort_param='stored', book_id=entry.series[0].id)|escapedlink(entry.series[0].name))|safe) }}

{% endif %} diff --git a/cps/templates/feed.xml b/cps/templates/feed.xml index fb9166d7..eed76051 100644 --- a/cps/templates/feed.xml +++ b/cps/templates/feed.xml @@ -70,7 +70,7 @@ {% endif %} {% for format in entry.Books.data %} + length="{{format.uncompressed_size}}" title="{{format.format}}" mtime="{{entry.Books.atom_timestamp}}" type="{{format.format|lower|mimetype}}"/> {% endfor %} {% endfor %} diff --git a/cps/templates/http_error.html b/cps/templates/http_error.html index 26c3c379..1c3e17a8 100644 --- a/cps/templates/http_error.html +++ b/cps/templates/http_error.html @@ -48,7 +48,11 @@
{% if g.current_theme == 1 %} -
+
{% endif %} {% if current_user.is_authenticated or g.allow_anonymous %} - {% endif %} {% if entry.series|length > 0 %} -

{{_("Book %(index)s of %(range)s", index=entry.series_index | formatfloat(2), range=(url_for('web.books_list', data='series', sort_param='stored', book_id=entry.series[0].id)|escapedlink(entry.series[0].name))|safe)}}

+

{{_("Book %(index)s of %(range)s", index=entry.series_index|formatfloat(2), range=(url_for('web.books_list', data='series', sort_param='stored', book_id=entry.series[0].id)|escapedlink(entry.series[0].name))|safe)}}

{% endif %} diff --git a/cps/templates/locales/en/translation.json b/cps/templates/locales/en/translation.json deleted file mode 100644 index 655669ee..00000000 --- a/cps/templates/locales/en/translation.json +++ /dev/null @@ -1,10 +0,0 @@ -{ - "input": { - "placeholder": "a placeholder" - }, - "nav": { - "home": "Home", - "page1": "Page One", - "page2": "Page Two" - } -} \ No newline at end of file diff --git a/cps/templates/read.html b/cps/templates/read.html index 1ca6a831..6f2ae99a 100644 --- a/cps/templates/read.html +++ b/cps/templates/read.html @@ -77,7 +77,7 @@

{{_('Settings')}}

- Choose a theme below:
+ {{_('Choose a theme below:')}}}
- - + +
{% if cc|length > 0 %} diff --git a/cps/templates/shelf.html b/cps/templates/shelf.html index bf0b1ab4..0c25b2cd 100644 --- a/cps/templates/shelf.html +++ b/cps/templates/shelf.html @@ -15,16 +15,19 @@ {% if entries.__len__() %} {{ _('Arrange books manually') }} - + {% endif %} {% endif %} @@ -67,7 +70,7 @@ {{entry.Books.series[0].name}} - ({{entry.Books.series_index|formatseriesindex}}) + ({{entry.Books.series_index|formatfloat(2)}})

{% endif %} {% if entry.Books.ratings.__len__() > 0 %} diff --git a/cps/templates/shelf_order.html b/cps/templates/shelf_order.html index ee90fd47..5ec7a4b1 100644 --- a/cps/templates/shelf_order.html +++ b/cps/templates/shelf_order.html @@ -20,7 +20,7 @@ {{entry['Books']['title']}} {% if entry['Books']['series']|length > 0 %}
- {{entry['Books']['series_index']}} - {{entry['Books']['series'][0].name}} + {{entry['Books']['series_index']|formatfloat(2)}} - {{entry['Books']['series'][0].name}} {% endif %}
{% for author in entry['Books']['author'] %} diff --git a/cps/templates/shelfdown.html b/cps/templates/shelfdown.html index 19a42b01..ecf5b252 100644 --- a/cps/templates/shelfdown.html +++ b/cps/templates/shelfdown.html @@ -45,7 +45,7 @@ {{entry.Books.series[0].name}} - ({{entry.Books.series_index}}) + ({{entry.Books.series_index|formatfloat(2)}})

{% endif %}
@@ -55,7 +55,7 @@ {% if entry.Books.data|length %}
{% for format in entry.Books.data %} - + {{format.format}} ({{ format.uncompressed_size|filesizeformat }}) {% endfor %} diff --git a/cps/templates/user_edit.html b/cps/templates/user_edit.html index bbb7a70d..b24c6ec0 100644 --- a/cps/templates/user_edit.html +++ b/cps/templates/user_edit.html @@ -25,7 +25,7 @@
{% endif %}
- +
{% if not content.role_anonymous() %} diff --git a/cps/translations/cs/LC_MESSAGES/messages.mo b/cps/translations/cs/LC_MESSAGES/messages.mo index 0ef2c43c..493d8567 100644 Binary files a/cps/translations/cs/LC_MESSAGES/messages.mo and b/cps/translations/cs/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/cs/LC_MESSAGES/messages.po b/cps/translations/cs/LC_MESSAGES/messages.po index d9bc237a..3f29168a 100644 --- a/cps/translations/cs/LC_MESSAGES/messages.po +++ b/cps/translations/cs/LC_MESSAGES/messages.po @@ -6,7 +6,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2020-06-09 21:11+0100\n" "Last-Translator: Lukas Heroudek \n" "Language: cs_CZ\n" @@ -15,512 +15,530 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistika" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Server restartován, znovu načtěte stránku" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Vypínám server, zavřete okno" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Neznámý příkaz" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Kniha byla úspěšně zařazena do fronty pro odeslání na %(eReadermail)s" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Neznámý" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Stránka správce" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Základní konfigurace" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Konfigurace uživatelského rozhraní" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Vlastní sloupec %(column)d neexistuje v databázi" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Uživatel admin" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Vše" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Uživatel nenalezen" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Zobrazit vše" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Nezbývá žádný správce, nelze odebrat roli správce" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Konfigurace Calibre-Web aktualizována" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Opravdu chcete odstranit Kobo token?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Jste si jisti, že chcete odstranit tuto polici?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Jste si jisti, že chcete odstranit tuto polici?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Jste si jisti, že chcete odstranit tuto polici?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Jste si jisti, že chcete odstranit tuto polici?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Opravdu chcete vypnout?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Zakázat" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Povolit" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json není nakonfigurováno pro webové aplikace" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Umístění zápisového souboru není platné. Určete prosím platnou polohu" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Umístění zápisového souboru pro přístup není platné. Určete prosím platnou polohu" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Prosím zadejte LDAP poskytovatele, port, DN a Identifikátor objektu uživatele" -#: cps/admin.py:1211 +#: cps/admin.py:1223 #, fuzzy msgid "Please Enter a LDAP Service Account and Password" msgstr "Zadejte platné uživatelské jméno pro obnovení hesla" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Filtr objektů skupiny LDAP musí mít jeden “%s” formátový identifikátor" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Filtr objektů skupiny LDAP má nesrovnatelnou závorku" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Filtr uživatelských objektů LDAP musí mít jeden “%s” formátový identifikátor" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Filtr uživatelských objektů LDAP má nesrovnatelnou závorku" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Přidat nového uživatele" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Změnit SMTP nastavení" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Chyba databáze: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Při odesílání zkušebního e-mailu došlo k chybě: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Prvně nastavte svou e-mailovou adresu..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Nastavení e-mailového serveru aktualizováno" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Neznámá chyba. Opakujte prosím později." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Upravit uživatele %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Heslo pro uživatele %(user)s resetováno" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Nejprve nakonfigurujte nastavení pošty SMTP..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Prohlížeč log souborů" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Požadování balíčku aktualizace" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Stahování balíčku aktualizace" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Rozbalování balíčku aktualizace" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Nahrazování souborů" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Databázová připojení jsou uzavřena" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Zastavuji server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Aktualizace dokončena, klepněte na tlačítko OK a znovu načtěte stránku" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Aktualizace selhala:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP chyba" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Chyba připojení" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Vypršel časový limit při navazování spojení" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Všeobecná chyba" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Aktualizační soubor nemohl být uložen do Temp Dir" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Nepodařilo se vytvořit nejméně jednoho uživatele LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Nepodařilo se vytvořit nejméně jednoho uživatele LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Chyba: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Chyba: Žádná reakce od uživatele LDAP serveru" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Nejméně jeden uživatel LDAP nenalezen v databázi" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Umístění databáze není platné, opravte prosím cestu" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Databáze není zapisovatelná" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Umístění souboru klíčů není platné, zadejte prosím správnou cestu" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Umístění certifikátu není platné, zadejte prosím správnou cestu" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Nastavení e-mailového serveru aktualizováno" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Konfigurace funkcí" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Vyplňte všechna pole!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-mail není z platné domény" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Přidat nového uživatele" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Uživatel '%(user)s' vytvořen" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Byl nalezen existující účet pro tuto e-mailovou adresu nebo přezdívku." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Uživatel '%(nick)s' smazán" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Nezbývá žádný správce, nemůžete jej odstranit" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Uživatel '%(nick)s' aktualizován" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Hledat" + #: cps/converter.py:31 msgid "not installed" msgstr "není nainstalováno" @@ -529,128 +547,123 @@ msgstr "není nainstalováno" msgid "Execution permissions missing" msgstr "Chybí povolení k exekuci" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Vlastní sloupec %(column)d neexistuje v databázi" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Žádné" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Jejda! Vybraná kniha není k dispozici. Soubor neexistuje nebo není přístupný" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadata úspěšně aktualizována" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Soubor %(file)s nahrán" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Chybí zdrojový nebo cílový formát pro převod" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Kniha byla úspěšně zařazena do fronty pro převod do %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Při převodu této knihy došlo k chybě: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Nahraná kniha pravděpodobně existuje v knihovně, zvažte prosím změnu před nahráním nové: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Jejda! Vybraná kniha není k dispozici. Soubor neexistuje nebo není přístupný" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s není platným jazykem" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadata úspěšně aktualizována" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Nahraná kniha pravděpodobně existuje v knihovně, zvažte prosím změnu před nahráním nové: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Soubor s příponou '%(ext)s' nelze odeslat na tento server" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Soubor s příponou '%(ext)s' nelze odeslat na tento server" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Soubor, který má být odeslán musí mít příponu" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Soubor %(filename)s nemohl být uložen do dočasného adresáře" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Nepodařilo se přesunout soubor obalu %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Formát knihy úspěšně smazán" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Kniha úspěšně smazána" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "upravit metadata" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Nepodařilo se vytvořit cestu %(path)s (oprávnění odepřeno)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Uložení souboru %(file)s se nezdařilo." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Formát souboru %(ext)s přidán do %(book)s" @@ -663,486 +676,479 @@ msgstr "Google Drive nastavení nebylo dokončeno, zkuste znovu deaktivovat a a msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Doména zpětného volání není ověřena, postupujte podle pokynů k ověření domény v konzole pro vývojáře google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s formát pro knihu: %(book)d nenalezen" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s nenalezen na Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s nenalezen: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Poslat do Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Tento e-mail byl odeslán přes Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web zkušební e-mail" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Zkušební e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Začínáme s Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Registrační e-mail pro uživatele: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Převést %(orig)s do %(format)s a poslat do Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Poslat %(format)s do Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Poslat do Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Požadovaný soubor nelze přečíst. Možná nesprávná oprávnění?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Mazání knihy selhalo %(id)s failed: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, fuzzy, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Mazání knihy %(id)s, cesta ke knize není platná %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Přejmenování názvu z: '%(src)s' na '%(dest)s' selhalo chybou: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Soubor %(file)s nenalezen na Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Přejmenování názvu z: '%(src)s' na '%(dest)s' selhalo chybou: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Cesta ke knize %(path)s nebyla nalezena na Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Zadané uživatelské jméno je již použito" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Chyba stahování obalu" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Chyba formátu obalu" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Vytvoření cesty obalu selhalo" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Soubor obalu není platný, nebo nelze uložit" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Pouze jpg/jpeg jsou podporované soubory pro obal" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Objevte" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "UnRar binární soubor nenalezen" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Chyba provádění UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "Databáze není zapisovatelná" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Chybí povolení k exekuci" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Chyba provádění UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Pro získání platného api_endpoint pro zařízení Kobo, přístupte na calibre-web bez localhost" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo nastavení" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Registrovat s %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "nyní jste přihlášen jako: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Připojení k %(oauth)s úspěšné" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Přihlášení selhalo, žádný uživatel s OAuth účtem" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Odpojení od %(oauth)s úspěšné" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Odpojení od %(oauth)s selhalo" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Přihlášení pomocí GitHub selhalo." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Nepodařilo se načíst informace o uživateli z GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Přihlášení pomocí Google selhalo." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Nepodařilo se načíst informace o uživateli z Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth chyba, prosím opakujte později." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth chyba, prosím opakujte později." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Přihlásit" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token nenalezen" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token vypršel" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Úspěch! Vraťte se prosím do zařízení" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Knihy" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Zobrazit nedávné knihy" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Žhavé knihy" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Zobrazit žhavé knihy" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Nejlépe hodnocené knihy" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Zobrazit nejlépe hodnocené knihy" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Přečtené knihy" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Zobrazit prečtené a nepřečtené" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Nepřečtené knihy" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Zobrazit nepřečtené" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Objevte" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Zobrazit náhodné knihy" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategorie" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Zobrazit výběr kategorie" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Série" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Zobrazit výběr sérií" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autoři" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Zobrazit výběr autora" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Vydavatelé" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Zobrazit výběr vydavatele" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Jazyky" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Zobrazit výběr jazyka" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Hodnocení" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Zobrazit výběr hodnocení" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formáty souborů" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Zobrazit výběr formátů" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Archivované knihy" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Zobrazit archivované knihy" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Hledat" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Vydáno po " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Vydáno před " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Hodnocení <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Hodnocení >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Rozšířené hledání" @@ -1198,7 +1204,7 @@ msgstr "Kniha byla odebrána z police: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Vytvořit polici" @@ -1253,45 +1259,45 @@ msgstr "Veřejná police s názvem '%(title)s' již existuje." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Osobní police s názvem ‘%(title)s’ již existuje." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Police: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Chyba otevírání police. Police neexistuje nebo není přístupná" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Úlohy" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Čekám" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Selhalo" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Spuštěno" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Dokončeno" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Neznámý stav" @@ -1324,178 +1330,178 @@ msgstr "Nová aktualizace k dispozici. Klepnutím na tlačítko níže aktualizu msgid "No release information available" msgstr "Nejsou k dispozici žádné informace o verzi" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Objevte (Náhodné knihy)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Žhavé knihy (Nejstahovanější)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Autoři: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Vydavatel: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Série: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Hodnocení: %(rating)s stars" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Soubor formátů: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategorie: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Jazyky: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Stáhnutí" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Seznam hodnocení" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Seznam formátů" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Nejprve nakonfigurujte nastavení pošty SMTP..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Kniha byla úspěšně zařazena do fronty pro odeslání na %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Při odesílání této knihy došlo k chybě: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Nejprve nakonfigurujte vaši kindle e-mailovou adresu.." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registrovat" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "E-mailový server není nakonfigurován, kontaktujte svého správce!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "E-mailový server není nakonfigurován, kontaktujte svého správce!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Váš e-mail nemá povolení k registraci" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Potvrzovací e-mail byl odeslán na váš účet." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Nelze aktivovat ověření LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "nyní jste přihlášen jako: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Záložní přihlášení jako: ‘%(nickname)s’, server LDAP není dosažitelný nebo neznámý uživatel" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Nelze se přihlásit: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Špatné uživatelské jméno nebo heslo" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Nové heslo bylo zasláno na vaši emailovou adresu" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Neznámá chyba. Opakujte prosím později." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Zadejte platné uživatelské jméno pro obnovení hesla" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "nyní jste přihlášen jako: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s profil" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profil aktualizován" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Byl nalezen existující účet pro tuto e-mailovou adresu." @@ -1504,54 +1510,58 @@ msgstr "Byl nalezen existující účet pro tuto e-mailovou adresu." msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Poslat do Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre převaděč %(tool)s nenalezen" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-převaděč selhal: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Převedený soubor nebyl nalezen nebo více než jeden soubor ve složce %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Převaděč eknih selhal: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Převaděč eknih selhal: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1560,30 +1570,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "upravit metadata" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Nahrávat" @@ -1602,12 +1608,12 @@ msgstr "Přezdívka" msgid "Email" msgstr "E-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Poslat do Kindle e-mailová adresa" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Správce" @@ -1617,8 +1623,8 @@ msgstr "Správce" msgid "Password" msgstr "Heslo" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Stahovat" @@ -1829,13 +1835,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Zrušit" @@ -1885,16 +1891,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "redukovat" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Více od" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Jazyk" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Vydavatel" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publikováno" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Popis:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Předchozí" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Další" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Nenalezeny žádné výsledky" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Domů" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Hledat v knihovně" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Odhlásit se" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Smazat knihu" @@ -1923,99 +1989,107 @@ msgstr "Převést do:" msgid "Convert book" msgstr "Převést knihu" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Nahrávání..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Zavřít" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Chyba" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Nahrávání hotovo, zpracovávám, čekejte prosím..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Nahrát formát" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Název knihy" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Štítky" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID série" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Datum vydání" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Hodnocení" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Popis" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identifikátory" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Typy identifikátorů" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Hodnota identifikátorů" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Odstranit" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Přidat identifikátor" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Štítky" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID série" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Hodnocení" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Adresa URL obalu (jpg, obal je stažen a uložen v databázi, pole je potom opět prázdné)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Nahrát obal z místní jednotky" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Datum vydání" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Vydavatel" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Jazyk" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ano" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Ne" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Nahrát formát" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Zobrazit knihu po uložení" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Získat metadata" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2023,38 +2097,32 @@ msgstr "Získat metadata" msgid "Save" msgstr "Uložit" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Klíčové slovo" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr "Hledat klíčové slovo" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Kliknutím na obal načtěte metadata do formuláře" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Načítání..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Zavřít" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Zdroj" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Chyba vyhledávání!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nebyly nalezeny žádné výsledky! Zadejte jiné klíčové slovo." @@ -2163,7 +2231,7 @@ msgid "Enter " msgstr "Identifikátory" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Jste si opravdu jisti?" @@ -2642,74 +2710,61 @@ msgstr "Přidat povolené/zakázané štítky" msgid "Add Allowed/Denied custom column values" msgstr "Přidat povolené/zakázané hodnoty vlastních sloupců" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Číst v prohližeči" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Poslechnout v prohlížeči" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publikováno" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Označit jako nepřečtené" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Označit jako přečtené" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Označit jako nepřečtené" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Přečteno" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Obnovit z archivu" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Archívovat" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Archivováno" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Popis:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Přidat do police" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Veřejné)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Upravit metadata" @@ -2781,10 +2836,6 @@ msgstr "Zadejte jméno domény" msgid "Denied Domains (Blacklist)" msgstr "Zakázané domény pro registraci" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Další" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Otevřte soubor .kobo/Kobo/Kobo eReader.conf v textovém editoru a vložte (nebo upravte):" @@ -2807,11 +2858,16 @@ msgstr "E-mailový server není nakonfigurován, kontaktujte svého správce!" msgid "Create Issue" msgstr "Vytvořit problém" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Konfigurace funkcí" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Zpět domů" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2901,7 +2957,7 @@ msgstr "Knihy řazené podle hodnocení" msgid "Books ordered by file formats" msgstr "Knihy seřazené podle souboru formátů" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Police" @@ -2910,60 +2966,37 @@ msgstr "Police" msgid "Books organized in shelves" msgstr "Knihy organizované v policích" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Domů" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Přepnout navigaci" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Hledat v knihovně" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Jednoduché" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Účet" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Odhlásit se" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Nahrávání..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Chyba" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Nahrávání hotovo, zpracovávám, čekejte prosím..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Nastavení" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Prosím neobnovujte stránku" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Procházet" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "O knihovně" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Předchozí" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Podrobnosti o knize" @@ -3079,7 +3112,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Kniha" @@ -3088,36 +3121,81 @@ msgstr "Kniha" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web katalog eknih" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "Čtečka PDF" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Zvolte uživatelské jméno" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Světlý" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Tmavý" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Zpět" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Po otevření postranních panelů přeformátujte text." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Smazat" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Čekám" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Svisle" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Přečteno" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3299,10 +3377,6 @@ msgstr "Tento ověřovací odkaz vyprší za 10 minut." msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Nenalezeny žádné výsledky" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Termín vyhledávání:" @@ -3319,11 +3393,11 @@ msgstr "Datum vydání od" msgid "Published Date To" msgstr "Datum vydání do" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3360,11 +3434,13 @@ msgstr "Hodnoceni více než" msgid "Rating Below" msgstr "Hodnocení méně než" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3388,6 +3464,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Sdílet se všemi" @@ -3456,15 +3540,19 @@ msgstr "Průběh" msgid "Run Time" msgstr "Doba spuštění" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3472,6 +3560,10 @@ msgstr "" msgid "Reset user Password" msgstr "Resetovat uživatelské heslo" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Zobrazit knihy s jazykem" diff --git a/cps/translations/de/LC_MESSAGES/messages.mo b/cps/translations/de/LC_MESSAGES/messages.mo index 0c47e64c..f5e999b4 100644 Binary files a/cps/translations/de/LC_MESSAGES/messages.mo and b/cps/translations/de/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/de/LC_MESSAGES/messages.po b/cps/translations/de/LC_MESSAGES/messages.po index 3da542fb..197d777d 100644 --- a/cps/translations/de/LC_MESSAGES/messages.po +++ b/cps/translations/de/LC_MESSAGES/messages.po @@ -7,8 +7,8 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" -"PO-Revision-Date: 2023-10-21 15:45+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-11-16 20:41+0100\n" "Last-Translator: Ozzie Isaacs\n" "Language: de\n" "Language-Team: \n" @@ -16,627 +16,639 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistiken" -#: cps/admin.py:150 +#: cps/admin.py:151 msgid "Server restarted, please reload page." msgstr "Server neu gestartet, Seite bitte neu laden." -#: cps/admin.py:152 +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." msgstr "Server wird heruntergefahren, Fenster bitte schließen." -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "Datenbank wurde erneut verbunden" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Unbekannter Befehl" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Bücher wurden für Metadaten Backup eingereiht, für das Ergebnis bitte Aufgaben überprüfen" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Unbekannt" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Admin Seite" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Basiskonfiguration" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Benutzeroberflächenkonfiguration" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Benutzerdefinierte Spalte Nr. %(column)d ist nicht in Calibre Datenbank vorhanden" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Benutzer bearbeiten" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Alle" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Benutzer nicht gefunden" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} Benutzer erfolgreich gelöscht" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Zeige alle" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Ungültige Anfrage" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" -msgstr "Guest Name kann nicht geändert werden" +msgstr "Name 'Guest' kann nicht geändert werden" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" -msgstr "Guest Benutzer kann diese Rolle nicht haben" +msgstr "Benutzer 'Guest' kann diese Rolle nicht haben" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" -msgstr "Kein Admin Benutzer verblieben Admin Berechtigung kann nicht entfernt werden" +msgstr "Kein Admin Benutzer verblieben, Admin Berechtigung kann nicht entfernt werden" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Wert muss true oder false sein" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Ungültige Rolle" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" -msgstr "Guest Benutzer kann diese Sichtbarkeit nicht haben" +msgstr "Benutzer 'Guest' kann diese Ansicht nicht haben" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" -msgstr "Ungültige Sichtbarkeit" +msgstr "Ungültige Ansicht" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" -msgstr "Guest Sprache wird automatisch bestimmt und kann nicht eingestellt werden" +msgstr "Sprache von 'Guest' wird automatisch bestimmt und kann nicht eingestellt werden" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Keine gültige Sprache gewählt" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Keine gültige Buchsprache gewählt" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parameter wurde nicht gefunden" -#: cps/admin.py:572 -msgid "Invalid Read Column" -msgstr "Ungültige Lese Spalte" - #: cps/admin.py:578 +msgid "Invalid Read Column" +msgstr "Ungültige Gelesen Spalte" + +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Ungültiger Spaltenname für Einschränkung" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Konfiguration von Calibre-Web wurde aktualisiert" -#: cps/admin.py:610 -msgid "Do you really want to delete the Kobo Token?" -msgstr "Möchten Sie wirklich den Kobo Token löschen?" - -#: cps/admin.py:612 -msgid "Do you really want to delete this domain?" -msgstr "Möchten Sie wirklich diese Domain löschen?" - -#: cps/admin.py:614 -msgid "Do you really want to delete this user?" -msgstr "Möchten Sie wirklich diesen Benutzer löschen?" - #: cps/admin.py:616 -msgid "Are you sure you want to delete this shelf?" -msgstr "Möchten Sie wirklich dieses Bücherregal löschen?" +msgid "Do you really want to delete the Kobo Token?" +msgstr "Möchten Sie den Kobo Token wirklich löschen?" #: cps/admin.py:618 -msgid "Are you sure you want to change locales of selected user(s)?" -msgstr "Möchten Sie wirklich die Anzeigesprache der ausgewählten Benutzer ändern?" +msgid "Do you really want to delete this domain?" +msgstr "Möchten Sie diese Domain wirklich löschen?" #: cps/admin.py:620 -msgid "Are you sure you want to change visible book languages for selected user(s)?" -msgstr "Möchten Sie wirklich die Büchersprachen für die ausgewählten Benutzer ändern?" +msgid "Do you really want to delete this user?" +msgstr "Möchten Sie diesen Benutzer wirklich löschen?" #: cps/admin.py:622 -msgid "Are you sure you want to change the selected role for the selected user(s)?" -msgstr "Möchten Sie wirklich die ausgewählte Rolle für die ausgewählten Benutzer verändern?" +msgid "Are you sure you want to delete this shelf?" +msgstr "Möchten Sie dieses Bücherregal wirklich löschen?" #: cps/admin.py:624 -msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" -msgstr "Möchten Sie wirklich die ausgewählten Sichtbarkeitsbeschränkungen der ausgewählten Benutzer ändern?" +msgid "Are you sure you want to change locales of selected user(s)?" +msgstr "Möchten Sie die Anzeigesprache der ausgewählten Benutzer wirklich ändern?" #: cps/admin.py:626 +msgid "Are you sure you want to change visible book languages for selected user(s)?" +msgstr "Möchten Sie die Büchersprachen für die ausgewählten Benutzer wirklich ändern?" + +#: cps/admin.py:628 +msgid "Are you sure you want to change the selected role for the selected user(s)?" +msgstr "Möchten Sie die ausgewählte Rolle für die ausgewählten Benutzer wirklich verändern?" + +#: cps/admin.py:630 +msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" +msgstr "Möchten Sie die ausgewählten Sichtbarkeitsbeschränkungen der ausgewählten Benutzer wirklich ändern?" + +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" -msgstr "Möchten Sie wirklich die Sichtbarkeiten für die ausgewählten Benutzer verändern?" +msgstr "Möchten Sie die Sichtbarkeiten für die ausgewählten Benutzer wirklich verändern?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" -msgstr "Möchten Sie wirklich die Synchronisation von Bücherregalen für die ausgewählten Benutzer verändern?" +msgstr "Möchten Sie die Synchronisation von Bücherregalen für die ausgewählten Benutzer wirklich verändern?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Ort der Calibre Datenbank editieren?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calibre-Web wird nach neuen Covern suchen und Cover Miniaturansichten aktualisieren, dies kann eine Weile dauern?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" -msgstr "Möchten Sie wirklich die Synchronisationsdatenbank von Calibre-Web löschen, um eine komplette Synchronisation zu erzwingen?" +msgstr "Möchten Sie die Synchronisationsdatenbank von Calibre-Web wirklich löschen, um eine komplette Synchronisation zu erzwingen?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Verbieten" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Erlauben" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} Synchronisationseinträge gelöscht" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Tag nicht gefunden" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Ungültige Aktion" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json ist nicht für Web Anwendungen konfiguriert" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Logdatei Pfad ist ungültig, bitte einen gültigen Pfad angeben" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" -msgstr "Zugriffs Logdatei Pfad ist ungültig, bitte einen gültigen Pfad angeben" +msgstr "Zugriffs-Logdatei Pfad ist ungültig, bitte einen gültigen Pfad angeben" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" -msgstr "Bitte einen LDAP Server, Port, DN und Benutzer Objekt angeben" +msgstr "Bitte einen LDAP Server, Port, DN und Benutzer Objekt Identifikator angeben" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" -msgstr "Bitte einen LDAP Service Account und Password eingeben" +msgstr "Bitte einen LDAP Service Account und Passwort eingeben" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Bitte einen LDAP Service Account eingeben" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP Gruppen Objekt Filter benötigt genau eine \"%s\" Format Kennung" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAP Gruppen Objekt Filter hat ungleiche Anzahl von Klammern" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP Benutzer Objekt Filter benötigt genau eine \"%s\" Format Kennung" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAP Benutzer Objekt Filter hat ungleiche Anzahl von Klammern" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Der LDAP Member User Filter benötigt genau eine \"%s\" Formatierungsmarkierung" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "LDAP Member User Filter hat eine ungleiche Anzahl von geöffneten und geschlossenen Klammern" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CA-Zertifikat, Zertifikat oder Key Datei ist kein gültiger Pfad" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Neuen Benutzer hinzufügen" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "SMTP-Einstellungen ändern" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "G-Mail Konto verifiziert." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Datenbankfehler: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Test E-Mail an %(email)s wurde zum Senden in die Warteschlange eingereiht, für das Ergebnis bitte Aufgaben überprüfen" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Es trat ein Fehler beim Versenden der Test-E-Mail auf: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Bitte zuerst E-Mail Adresse konfigurieren..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Einstellungen des E-Mail-Servers aktualisiert" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Einstellungen für Geplante Aufgaben" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Ungültigen Startzeitpunkt für Aufgaben spezifiziert" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Ungültige Laufzeit für Aufgaben spezifiziert" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Einstellungen für Geplante Aufgaben aktualisiert" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Es ist ein unbekannter Fehler aufgetreten. Bitte später erneut versuchen." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Einstellungsdatenbank ist nicht schreibbar" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Benutzer %(nick)s bearbeiten" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "Passwort für Benutzer %(user)s wurde zurückgesetzt" -#: cps/admin.py:1451 +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." -msgstr "Bitte zuerst die SMTP-Einstellung konfigurieren." +msgstr "Bitte zuerst die SMTP-Einstellungen konfigurieren." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Logdatei Anzeige" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Frage Update an" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Lade Update herunter" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Entpacke Update" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Ersetze Dateien" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Schließe Datenbankverbindungen" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Stoppe Server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Update abgeschlossen, bitte okay drücken und Seite neu laden" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Update fehlgeschlagen:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP Fehler" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Verbindungsfehler" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Timeout beim Verbindungsaufbau" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Allgemeiner Fehler" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" -msgstr "Updatedatei konnte nicht in Temporärem Ordner gespeichert werden" +msgstr "Updatedatei konnte nicht in temporärem Ordner gespeichert werden" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Dateien konnten während des Updates nicht ausgetauscht werden" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Mindestens ein LDAP Benutzer konnte nicht extrahiert werden" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Mindestens ein LDAP Benutzer konnte nicht erzeugt werden" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Fehler: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Fehler: Keine Benutzerinformationen von LDAP Server empfangen" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Mindestens ein LDAP Benutzer wurde nicht in der Datenbank gefudnen" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Benutzer erfolgreich importiert" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "Bücherpfad ungültig" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "DB Pfad ist nicht gültig, bitte einen gültigen Pfad angeben" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Datenbank ist nicht schreibbar" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Schlüsseldatei ist ungültig, bitte einen gültigen Pfad angeben" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Zertifikatsdatei ist ungültig, bitte einen gültigen Pfad angeben" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "Passwortlänge muss zwischen 1 und 40 Zeichen liegen" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Datenbankeinstellung aktualisiert" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Datenbank-Konfiguration" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Bitte alle Felder ausfüllen." -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-Mail bezieht sich nicht auf eine gültige Domain" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Neuen Benutzer hinzufügen" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Benutzer '%(user)s' angelegt" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." -msgstr "Es existiert bereits ein Account für diese E-Mailadresse oder diesen Benutzernamen." +msgstr "Es existiert bereits ein Account für diese E-Mail Adresse oder diesen Benutzernamen." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Benutzer '%(nick)s' gelöscht" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" -msgstr "Guest Benutzer kann nicht gelöscht werden" +msgstr "Benutzer 'Guest' kann nicht gelöscht werden" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Benutzer kann nicht gelöscht werden, es wäre kein Admin Benutzer übrig" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "E-Mail kann nicht leer sein und muss gültig sein" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Benutzer '%(nick)s' aktualisiert" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Suchen" + #: cps/converter.py:31 msgid "not installed" msgstr "Nicht installiert" #: cps/converter.py:32 msgid "Execution permissions missing" -msgstr "Ausführeberechtigung fehlt" +msgstr "Ausführberechtigung fehlt" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Benutzerdefinierte Spalte Nr. %(column)d ist nicht in Calibre Datenbank vorhanden" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Keine" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Öffnen des Buchs fehlgeschlagen. Datei existiert nicht oder ist nicht zugänglich" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Benutzer hat keine Berechtigung Cover hochzuladen" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "IDs unterscheiden nicht Groß-Kleinschreibung, alte ID wird überschrieben" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadaten wurden erfolgreich aktualisiert" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Fehler beim editieren des Buches: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Datei %(file)s hochgeladen" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Quell- oder Zielformat für Konvertierung fehlt" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Buch wurde erfolgreich für die Konvertierung nach %(book_format)s eingereiht" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Es trat ein Fehler beim Konvertieren des Buches auf: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Das hochgeladene Buch existiert evtl. schon in der Bibliothek: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Öffnen des Buches fehlgeschlagen. Datei existiert nicht oder ist nicht zugänglich" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Benutzer hat keine Berechtigung Cover hochzuladen" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "IDs unterscheiden nicht zwischen Groß- und Kleinschreibung, alte ID wird überschrieben" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "'%(langname)s' ist keine gültige Sprache" -#: cps/editbooks.py:756 cps/editbooks.py:1192 -#, fuzzy -msgid "File type isn't allowed to be uploaded to this server" -msgstr "Dateiendung '%(ext)s' kann nicht auf diesen Server hochgeladen werden" +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadaten wurden erfolgreich aktualisiert" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Fehler beim Editieren des Buches: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Das hochgeladene Buch existiert evtl. schon in der Bibliothek: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 +msgid "File type isn't allowed to be uploaded to this server" +msgstr "Dateityp kann nicht auf diesen Server hochgeladen werden" + +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Dateiendung '%(ext)s' kann nicht auf diesen Server hochgeladen werden" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Dateien müssen eine Erweiterung haben, um hochgeladen zu werden" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Die Datei %(filename)s konnte nicht im temporären Ordner gespeichert werden" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Fehler beim Verschieben der Cover Datei %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Buch Format erfolgreich gelöscht" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Buch erfolgreich gelöscht" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" -msgstr "Keine Erlaubnis zum Bücher löschen" +msgstr "Keine Erlaubnis zum Löschen von Büchern" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "Metadaten editieren" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" -msgstr "%(seriesindex)s ist keine gültige Zahl, Eintrag wird ignoriert" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "Serienindex %(seriesindex)s ist keine gültige Zahl, Eintrag wird ignoriert" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Benutzer hat kein Recht zusätzliche Dateiformate hochzuladen" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Fehler beim Erzeugen des Pfads %(path)s (Zugriff verweigert)" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Fehler beim Speichern der Datei %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Dateiformat %(ext)s zu %(book)s hinzugefügt" @@ -649,470 +661,461 @@ msgstr "Google Drive Setup is nicht komplett, bitte versuche Google Drive zu dea msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Callback Domain ist nicht verifiziert, bitte Domain in der Google Developer Console verifizieren" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s Format für Buch-ID %(book)d nicht gefunden" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s von Buch %(fn)s nicht auf Google Drive gefunden" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s nicht gefunden: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "An E-Reader senden" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "Diese E-Mail wurde durch Calibre-Web versendet." -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "Calibre-Web Test-E-Mail" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "Test E-Mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Loslegen mit Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "Registrierungs-E-Mail für Benutzer: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Konvertiere %(orig)s nach %(format)s und sende an E-Reader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, python-format msgid "Send %(format)s to eReader" msgstr "Sende %(format)s an E-Reader" -#: cps/helper.py:227 +#: cps/helper.py:230 #, python-format msgid "%(book)s send to eReader" msgstr "%(book)s an E-Reader gesendet" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Die angeforderte Datei konnte nicht gelesen werden. Evtl. falsche Zugriffsrechte?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" -msgstr "Gelesenen Status konnte nicht aktualisiert werden: {}" +msgstr "Gelesen-Status konnte nicht aktualisiert werden: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Löschen des Ordners für Buch %(id)s ist fehlgeschlagen, der Pfad hat Unterordner: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Löschen von Buch %(id)s fehlgeschlagen: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Lösche Buch %(id)s nur aus Datenbank, Pfad zum Buch in Datenbank ist nicht gültig: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "Umbenennen des Autors '%(src)s' zu '%(dest)s' schlug fehl: %(error)s" +msgstr "Umbenennen des Autors '%(src)s' zu '%(dest)s' fehlgeschlagen: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Datei %(file)s wurde nicht auf Google Drive gefunden" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "Umbenennen des Titels '%(src)s' zu '%(dest)s' schlug fehl: %(error)s" +msgstr "Umbenennen des Titels '%(src)s' zu '%(dest)s' fehlgeschlagen: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Buchpfad %(path)s wurde nicht auf Google Drive gefunden" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "Es existiert bereits ein Benutzerkonto für diese E-Mail Adresse" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Benutzername ist schon vorhanden" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "Ungültiges E-Mail Adressformat" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" -msgstr "Passwort stimmt nicht mit den Passwortregln überein" +msgstr "Passwort stimmt nicht mit den Passwortregeln überein" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" -msgstr "Python Module 'advocate' ist nicht installiert, wird aber für das Cover hochladen benötigt" +msgstr "Python Modul 'advocate' ist nicht installiert, wird aber für das Hochladen von Covern benötigt" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Fehler beim Herunterladen des Covers" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Coverdatei fehlerhaft" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" -msgstr "Keine Berechtigung Cover von Localhost oder dem lokalen Netzwerk hochzuladen" +msgstr "Keine Berechtigung Cover von localhost oder dem lokalen Netzwerk hochzuladen" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Fehler beim Erzeugen des Ordners für die Coverdatei" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" -msgstr "Cover Datei ist keine gültige Bilddatei, kann nicht gespeichert werden" +msgstr "Coverdatei ist keine gültige Bilddatei, kann nicht gespeichert werden" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Nur jpg/jpeg/png/webp/bmp Dateien werden als Coverdatei unterstützt" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" -msgstr "Ungültiger Cover Dateiinhalt" +msgstr "Ungültiger Coverdatei-Inhalt" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" -msgstr "Es werden nur jpg/jpeg Dateien als Cover untertützt" +msgstr "Es werden nur jpg/jpeg Dateien als Cover unterstützt" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" msgstr "Titelbild" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "UnRar Programm nicht gefunden" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "Fehler beim Ausführen von UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" -msgstr "" +msgstr "Angegebener Ordner konnte nicht gefunden werden" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" -msgstr "" +msgstr "Bitte keine Datei sondern einen Ordner angeben" -#: cps/helper.py:1064 -#, fuzzy +#: cps/helper.py:1042 msgid "Calibre binaries not viable" -msgstr "Datenbank ist nicht schreibbar" +msgstr "Calibre Programm ist nicht nutzbar" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" -msgstr "" +msgstr "Fehlende Calibre Binärdateien: %(missing)s" -#: cps/helper.py:1075 -#, fuzzy, python-format +#: cps/helper.py:1053 +#, python-format msgid "Missing executable permissions: %(missing)s" -msgstr "Ausführeberechtigung fehlt" +msgstr "Ausführberechtigung fehlt: %(missing)s" -#: cps/helper.py:1080 -#, fuzzy +#: cps/helper.py:1058 msgid "Error executing Calibre" -msgstr "Fehler beim Ausführen von UnRar" +msgstr "Fehler beim Ausführen von Calibre" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Alle Bücher für Metadaten Backup einreihen" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" -msgstr "Bitte nicht von \"localhost\" auf Calibre-Web zugreifen, um einen gültigen api_endpoint für Kobo Geräte zu erhalten" +msgstr "Bitte nicht von \"localhost\" auf Calibre-Web zugreifen, um einen gültigen api_endpoint für Kobo Geräte zu erhalten" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo Setup" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Anmelden mit %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Du bist nun eingeloggt als '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Verbindung mit %(oauth)s erfolgreich" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Login fehlgeschlagen, es ist kein Benutzer mit diesem Account verbunden" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Verbindung zu %(oauth)s erfolgreich getrennt" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Verbindung mit %(oauth)s fehlgeschlagen" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Nicht mit %(oauth)s verbunden" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." -msgstr "Login mit Github fehlgeschlagen." +msgstr "Login mit GitHub fehlgeschlagen." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." -msgstr "Laden der Benutzerinformationen von Github fehlgeschlagen." +msgstr "Laden der Benutzerinformationen von GitHub fehlgeschlagen." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Login mit Google fehlgeschlagen." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Laden der Benutzerinformationen von Google fehlgeschlagen." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." -msgstr "GitHub Oauth Fehler, bitte später erneut versuchen." +msgstr "GitHub OAuth Fehler, bitte später erneut versuchen." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" -msgstr "Github Oauth Fehler {}" +msgstr "Github OAuth Fehler {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." -msgstr "Google Oauth Fehler, bitte später erneut versuchen." +msgstr "Google OAuth Fehler, bitte später erneut versuchen." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" -msgstr "Google Oauth Fehler: {}" +msgstr "Google OAuth Fehler: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Sterne" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Login" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token wurde nicht gefunden" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token ist abgelaufen" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Erfolg! Bitte zum Gerät zurückkehren" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Bücher" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Zeige kürzlich hinzugefügte Bücher" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Beliebte Bücher" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Zeige beliebte Bücher" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Heruntergeladene Bücher" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Zeige heruntergeladene Bücher" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" -msgstr "Best bewertete Bücher" +msgstr "Am besten bewertete Bücher" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" -msgstr "Bestbewertete Bücher anzeigen" +msgstr "Zeige am besten bewertete Bücher" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Gelesene Bücher" -#: cps/render_template.py:64 +#: cps/render_template.py:63 msgid "Show Read and Unread" msgstr "Zeige gelesene/ungelesene Bücher" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Ungelesene Bücher" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" -msgstr "Zeige Ungelesene" +msgstr "Zeige ungelesene Bücher" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" -msgstr "Entdecke" +msgstr "Entdecken" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Zeige zufällige Bücher" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategorien" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" -msgstr "Zeige Kategorienauswahl" +msgstr "Zeige Kategorien" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Serien" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" -msgstr "Zeige Serienauswahl" +msgstr "Zeige Serien" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autoren" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" -msgstr "Zeige Autorenauswahl" +msgstr "Zeige Autoren" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Verleger" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" -msgstr "Zeige Verlegerauswahl" +msgstr "Zeige Verleger" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Sprachen" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" -msgstr "Zeige Sprachauswahl" +msgstr "Zeige Sprachen" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Bewertungen" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" -msgstr "Zeige Bewertungsauswahl" +msgstr "Zeige Bewertungen" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Dateiformate" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" -msgstr "Zeige Dateiformatauswahl" +msgstr "Zeige Dateiformate" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Archivierte Bücher" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "Zeige archivierte Bücher" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Bücherliste" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Zeige Bücherliste" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Suche" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Herausgegeben nach dem " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Herausgegeben vor dem " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Bewertung <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Bewertung >= %(rating)s" -#: cps/search.py:221 -#, fuzzy, python-format +#: cps/search.py:234 +#, python-format msgid "Read Status = '%(status)s'" -msgstr "Lesestatus = %(status)s" +msgstr "Gelesenstatus = '%(status)s'" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Fehler bei der Suche nach eigenen Spalten, bitte Calibre-Web neustarten" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Erweiterte Suche" @@ -1132,12 +1135,12 @@ msgstr "Buch ist bereits Teil des Bücherregals %(shelfname)s" #: cps/shelf.py:77 #, python-format msgid "%(book_id)s is a invalid Book Id. Could not be added to Shelf" -msgstr "" +msgstr "%(book_id)s ist eine ungültige Buch ID. Buch konnte nicht zu Bücherregal hinzugefügt werden" #: cps/shelf.py:97 #, python-format msgid "Book has been added to shelf: %(sname)s" -msgstr "Das Buch wurde dem Bücherregal %(sname)s hinzugefügt" +msgstr "Das Buch wurde zum Bücherregal %(sname)s hinzugefügt" #: cps/shelf.py:116 msgid "You are not allowed to add a book to the shelf" @@ -1167,13 +1170,13 @@ msgstr "Das Buch wurde aus dem Bücherregal: %(sname)s entfernt" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Sie haben keine Berechtigung um Bücher aus diesem Bücherregal zu löschen" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Bücherregal erzeugen" #: cps/shelf.py:226 msgid "Sorry you are not allowed to edit this shelf" -msgstr "Dir ist es nicht erlaubt, dieses Bücherregal zu editieren" +msgstr "Sie dürfen dieses Bücherregal nicht editieren" #: cps/shelf.py:228 msgid "Edit a shelf" @@ -1194,7 +1197,7 @@ msgstr "Reihenfolge in Bücherregal '%(name)s' verändern" #: cps/shelf.py:324 msgid "Sorry you are not allowed to create a public shelf" -msgstr "Sie haben keine Berechtigung um öffentliche Bücherregal zu erzeugen" +msgstr "Sie haben keine Berechtigung um ein öffentliches Bücherregal zu erzeugen" #: cps/shelf.py:341 #, python-format @@ -1213,52 +1216,52 @@ msgstr "Es trat ein Fehler auf" #: cps/shelf.py:380 #, python-format msgid "A public shelf with the name '%(title)s' already exists." -msgstr "Es existiert bereit ein öffentliches Bücherregal mit dem Name '%(title)s'." +msgstr "Es existiert bereit ein öffentliches Bücherregal mit dem Namen '%(title)s'." #: cps/shelf.py:391 #, python-format msgid "A private shelf with the name '%(title)s' already exists." -msgstr "Es existiert bereit ein privates Bücherregal mit dem Name '%(title)s'." +msgstr "Es existiert bereit ein privates Bücherregal mit dem Namen '%(title)s'." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Bücherregal: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Fehler beim Öffnen des Bücherregals. Bücherregal exisitert nicht oder ist nicht zugänglich" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Aufgaben" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Wartend" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Fehlgeschlagen" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Gestartet" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" -msgstr "Beendet" +msgstr "Fertiggestellt" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Beendet" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Abgebrochen" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Unbekannter Status" @@ -1291,225 +1294,228 @@ msgstr "Ein neues Update ist verfügbar. Klicke auf den Button unten, um auf Ver msgid "No release information available" msgstr "Keine Releaseinformationen verfügbar" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" -msgstr "Entdecke (Zufällige Bücher)" +msgstr "Entdecken (Zufällige Bücher)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" -msgstr "Beliebte Bücher (am meisten Downloads)" +msgstr "Beliebte Bücher (meiste Downloads)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Von %(user)s heruntergeladene Bücher" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Author: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Verleger: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Serie: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Bewertung: Keine" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Bewertung: %(rating)s Sterne" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Dateiformat: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategorie: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Sprache: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Downloads" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Bewertungsliste" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Liste der Dateiformate" -#: cps/web.py:1249 +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." msgstr "Bitte zuerst die SMTP-Einstellung konfigurieren..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Buch erfolgreich zum Senden an %(eReadermail)s eingereiht" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" -msgstr "Beim Senden des Buchs trat ein Fehler auf: %(res)s" +msgstr "Beim Senden des Buches trat ein Fehler auf: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Bitte zuerst die E-Reader E-Mailadresse konfigurieren." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" -msgstr "Bitte eine Minute warten vor der Registrierung des nächsten Benutzers " +msgstr "Bitte eine Minute warten vor der Registrierung des nächsten Benutzers" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registrieren" -#: cps/web.py:1281 cps/web.py:1385 -#, fuzzy +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" -msgstr "Der E-Mail Server ist nicht konfigurierte, bitte den Administrator kontaktieren." +msgstr "Verbindugnsfehler zu Limiter Backend, bitte Administrator kontaktieren" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." -msgstr "Der E-Mail Server ist nicht konfigurierte, bitte den Administrator kontaktieren." +msgstr "Der E-Mail Server ist nicht konfiguriert, bitte den Administrator kontaktieren." -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Diese E-Mail ist nicht für die Registrierung zugelassen." -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Eine Bestätigungs-E-Mail wurde an deinen E-Mail Account versendet." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "LDAP-Authentifizierung kann nicht aktiviert werden" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" -msgstr "Bitte eine Minute vor dem nächsten Loginversuche warten " +msgstr "Bitte eine Minute vor dem nächsten Loginversuch warten" -#: cps/web.py:1400 +#: cps/web.py:1408 #, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "Du bist nun eingeloggt als '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" -msgstr "Rückfall Login als: '%(nickname)s', LDAP Server ist nicht erreichbar, oder der Nutzer ist unbekannt" +msgstr "Fallback Login als: '%(nickname)s', LDAP Server ist nicht erreichbar, oder der Nutzer ist unbekannt" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "Login nicht erfolgreich: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" msgstr "Falscher Benutzername oder Passwort" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "Das neue Passwort wurde an die E-Mail Adresse verschickt" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "Es ist ein unbekannter Fehler aufgetreten. Bitte später erneut versuchen." -#: cps/web.py:1429 +#: cps/web.py:1437 msgid "Please enter valid username to reset password" msgstr "Bitte einen gültigen Benutzernamen zum Zurücksetzen des Passworts angeben" -#: cps/web.py:1437 +#: cps/web.py:1445 #, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Du bist nun eingeloggt als: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s's Profil" -#: cps/web.py:1511 +#: cps/web.py:1526 msgid "Success! Profile Updated" msgstr "Profil aktualisiert" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Es existiert bereits ein Benutzer für diese E-Mailadresse." #: cps/services/gmail.py:59 msgid "Found no valid gmail.json file with OAuth information" -msgstr "Keine gültige gmail.json Datei mit Oauth informationen gefunden" +msgstr "Keine gültige gmail.json Datei mit OAuth informationen gefunden" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "Temp Ordner Inhalt löschen" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s an E-Reader gesendet" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre E-Book Konverter %(tool)s nicht gefunden" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s Format nicht gefunden" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" -msgstr "EBook Converter mit unbekanntem Fehler fehlgeschlagen" +msgstr "E-Book Converter mit unbekanntem Fehler fehlgeschlagen" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify Konverter Aufruf fehlgeschlagen: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Konvertierte Datei nicht gefunden, oder mehr als eine Datei im Pfad %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Fehler des EBook-Converters: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre fehlgeschlagen mit Fehler: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Fehler des E-Book-Converters: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Konvertiere" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Calibre Datenbank wird neu verbunden" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "E-Mail" @@ -1517,30 +1523,26 @@ msgstr "E-Mail" msgid "Backing up Metadata" msgstr "Metadaten Backup läuft" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "%(count)s Cover Miniaturansichten erzeugt" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" -msgstr "Cover Miniaturansichtern" +msgstr "Cover Miniaturansichten" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "{0} Serien Miniaturansichten erzeugt" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Cover Miniaturansichten Cache wird gelöscht" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Upload" @@ -1559,11 +1561,11 @@ msgstr "Benutzername" msgid "Email" msgstr "E-Mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "An E-Reader E-Mail Adresse senden" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Admin" @@ -1573,8 +1575,8 @@ msgstr "Admin" msgid "Password" msgstr "Passwort" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Download" @@ -1634,7 +1636,7 @@ msgstr "E-Mail Service" #: cps/templates/admin.html:91 msgid "Gmail via Oauth2" -msgstr "Gmail via Oauth2" +msgstr "Gmail via OAuth2" #: cps/templates/admin.html:106 msgid "Configuration" @@ -1642,7 +1644,7 @@ msgstr "Konfiguration" #: cps/templates/admin.html:109 msgid "Calibre Database Directory" -msgstr "Ordner der Calibre-DB" +msgstr "Ordner der Calibre Datenbank" #: cps/templates/admin.html:113 cps/templates/config_edit.html:68 msgid "Log Level" @@ -1703,7 +1705,7 @@ msgstr "Geplante Aufgabe" #: cps/templates/admin.html:170 cps/templates/schedule_edit.html:12 #: cps/templates/tasks.html:18 msgid "Start Time" -msgstr "Zeitpunkt an dem die Aufgabe startet" +msgstr "Startzeit" #: cps/templates/admin.html:174 cps/templates/schedule_edit.html:20 msgid "Maximum Duration" @@ -1720,11 +1722,11 @@ msgstr "Seriencover Miniaturansichten erzeugen" #: cps/templates/admin.html:186 cps/templates/admin.html:208 #: cps/templates/schedule_edit.html:37 msgid "Reconnect Calibre Database" -msgstr "Mit Calibre Bibliothek neuverbinden" +msgstr "Mit Calibre Bibliothek neu verbinden" #: cps/templates/admin.html:190 cps/templates/schedule_edit.html:41 msgid "Generate Metadata Backup Files" -msgstr "Metadaten Backup Datei erzeugen " +msgstr "Metadaten Backup Datei erzeugen" #: cps/templates/admin.html:197 msgid "Refresh Thumbnail Cache" @@ -1784,13 +1786,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Abbruch" @@ -1813,12 +1815,12 @@ msgstr "In Bibliothek" #: cps/templates/author.html:26 cps/templates/index.html:74 #: cps/templates/search.html:31 cps/templates/shelf.html:20 msgid "Sort according to book date, newest first" -msgstr "Sortiere nach Buchdatum, Neuestes zuerst" +msgstr "Sortiere nach Buchdatum, neuestes zuerst" #: cps/templates/author.html:27 cps/templates/index.html:75 #: cps/templates/search.html:32 cps/templates/shelf.html:21 msgid "Sort according to book date, oldest first" -msgstr "Sortiere nach Buchdatum, Ältestes zuerst" +msgstr "Sortiere nach Buchdatum, ältestes zuerst" #: cps/templates/author.html:28 cps/templates/index.html:76 #: cps/templates/search.html:33 cps/templates/shelf.html:22 @@ -1833,23 +1835,83 @@ msgstr "Sortiere Titel in umgekehrt alphabetischer Reihenfolge" #: cps/templates/author.html:30 cps/templates/index.html:80 #: cps/templates/search.html:37 cps/templates/shelf.html:26 msgid "Sort according to publishing date, newest first" -msgstr "Sortiere nach Herausgabedatum, Neueste zuerst" +msgstr "Sortiere nach Herausgabedatum, neueste zuerst" #: cps/templates/author.html:31 cps/templates/index.html:81 #: cps/templates/search.html:38 cps/templates/shelf.html:27 msgid "Sort according to publishing date, oldest first" -msgstr "Sortiere nach Herausgabedatum, Älteste zuerst" +msgstr "Sortiere nach Herausgabedatum, älteste zuerst" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "Reduzieren" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Mehr von" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Buch %(index)s von %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Sprache" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Herausgeber" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Herausgabedatum" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Beschreibung:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Vorheriger Eintrag" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Nächste" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Keine Ergebnisse" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Home" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Bibliothek durchsuchen" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Logout" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Buch löschen" @@ -1860,7 +1922,7 @@ msgstr "Lösche Formate:" #: cps/templates/book_edit.html:25 msgid "Convert book format:" -msgstr "Konvertiere Buchformat:" +msgstr "Buchformat konvertieren:" #: cps/templates/book_edit.html:30 msgid "Convert from:" @@ -1876,101 +1938,109 @@ msgstr "Konvertiere nach:" #: cps/templates/book_edit.html:46 msgid "Convert book" -msgstr "Konvertiere Buch" +msgstr "Buch konvertieren" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Lade hoch..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Schließen" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Fehler" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Hochladen beendet, verarbeite Daten, bitte warten..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Format hochladen" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Buchtitel" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Tags" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Serien ID" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Herausgabedatum" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Bewertung" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Beschreibung" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "IDs" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "ID Typ" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "ID Wert" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Entfernen" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "ID hinzufügen" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Tags" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Serien ID" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Bewertung" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Cover-URL (jpg, Cover wird heruntergeladen und in der Datenbank gespeichert, Feld erscheint anschließend wieder leer)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" -msgstr "Coverdatei von Lokalem Laufwerk hochladen" +msgstr "Coverdatei von lokalem Laufwerk hochladen" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Herausgabedatum" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Herausgeber" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Sprache" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ja" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nein" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Format hochladen" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Buch nach Bearbeitung ansehen" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Metadaten laden" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -1978,37 +2048,31 @@ msgstr "Metadaten laden" msgid "Save" msgstr "Speichern" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Suchbegriff" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Suche Schlüsselbegriff" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Klicke auf das Bild, um die Metadaten zu übertragen" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Lade..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Schließen" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Quelle" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Fehler bei der Suche!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Keine Ergebnisse gefunden! Bitte ein anderes Schlüsselwort benutzen." @@ -2115,7 +2179,7 @@ msgid "Enter " msgstr "Eingeben " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Sicher?" @@ -2137,7 +2201,7 @@ msgstr "Speicherort der Calibre-Datenbank" #: cps/templates/config_db.html:21 msgid "Separate Book Files from Library" -msgstr "" +msgstr "Bücherdateien von Bibliotheksdatei separieren" #: cps/templates/config_db.html:34 msgid "Use Google Drive?" @@ -2153,7 +2217,7 @@ msgstr "Google Drive Calibre-Ordner" #: cps/templates/config_db.html:52 msgid "Metadata Watch Channel ID" -msgstr "Matadata Überwachungs-ID" +msgstr "Metadaten Überwachungs-ID" #: cps/templates/config_db.html:55 msgid "Revoke" @@ -2193,7 +2257,7 @@ msgstr "Nightly" #: cps/templates/config_edit.html:50 msgid "Trusted Hosts (Comma Separated)" -msgstr "Trusted Hosts (Komma separiert)" +msgstr "Trusted Hosts (kommasepariert)" #: cps/templates/config_edit.html:61 msgid "Logfile Configuration" @@ -2221,7 +2285,7 @@ msgstr "Nicht-englische Zeichen in Titel und Autor beim Speichern auf Festplatte #: cps/templates/config_edit.html:108 msgid "Embed Metadata to Ebook File on Download/Conversion/e-mail (needs Calibre/Kepubify binaries)" -msgstr "" +msgstr "Metadaten bei Download/Konvertierung/E-Mail zu Ebook Datei hinzufügen (benötigt Calibre/Kepubify)" #: cps/templates/config_edit.html:112 msgid "Enable Uploads" @@ -2229,7 +2293,7 @@ msgstr "Hochladen aktivieren" #: cps/templates/config_edit.html:112 msgid "(Please ensure that users also have upload permissions)" -msgstr "(Bitte stellen Sie sicher das sie über die Upload Berechtigung verfügen)" +msgstr "(Bitte stellen Sie sicher, dass Sie über die Upload Berechtigung verfügen)" #: cps/templates/config_edit.html:116 msgid "Allowed Upload Fileformats" @@ -2257,11 +2321,11 @@ msgstr "Synchronisation mit Kobo aktivieren" #: cps/templates/config_edit.html:146 msgid "Proxy unknown requests to Kobo Store" -msgstr "Unbekannte Anfragen an Kobo.com weiterleiten" +msgstr "Unbekannte Anfragen an kobo.com weiterleiten" #: cps/templates/config_edit.html:149 msgid "Server External Port (for port forwarded API calls)" -msgstr "Externer Server Port (für Port Weiterleitung von API Aufrufen)" +msgstr "Externer Server Port (für Portweiterleitung von API-Aufrufen)" #: cps/templates/config_edit.html:157 msgid "Use Goodreads" @@ -2411,9 +2475,8 @@ msgid "External binaries" msgstr "Externe Programme" #: cps/templates/config_edit.html:325 -#, fuzzy msgid "Path to Calibre Binaries" -msgstr "Pfad zum Calibre E-Book Konverter" +msgstr "Pfad zu Calibre Dateien" #: cps/templates/config_edit.html:333 msgid "Calibre E-Book Converter Settings" @@ -2424,7 +2487,6 @@ msgid "Path to Kepubify E-Book Converter" msgstr "Pfad zum Kepubify E-Book Konverter" #: cps/templates/config_edit.html:344 -#, fuzzy msgid "Location of Unrar binary" msgstr "Pfad zur UnRar-Binärdatei" @@ -2438,15 +2500,15 @@ msgstr "Login Fehlversuche begrenzen" #: cps/templates/config_edit.html:372 msgid "Configure Backend for Limiter" -msgstr "" +msgstr "Backend für Limiter konfigurieren" #: cps/templates/config_edit.html:376 msgid "Options for Limiter Backend" -msgstr "" +msgstr "Optionen für Limiter Backend" #: cps/templates/config_edit.html:382 msgid "Check if file extensions matches file content on upload" -msgstr "" +msgstr "Überprüfe bei Upload ob Dateiinhalt zur Endung passt" #: cps/templates/config_edit.html:385 msgid "Session protection" @@ -2462,7 +2524,7 @@ msgstr "Stark" #: cps/templates/config_edit.html:393 msgid "User Password policy" -msgstr "Passwort Regeln" +msgstr "Passwortregeln" #: cps/templates/config_edit.html:397 msgid "Minimum password length" @@ -2470,7 +2532,7 @@ msgstr "Minimale Passwortlänge" #: cps/templates/config_edit.html:402 msgid "Enforce number" -msgstr "Erzwinge Nummer" +msgstr "Erzwinge Zahl" #: cps/templates/config_edit.html:406 msgid "Enforce lowercase characters" @@ -2482,7 +2544,7 @@ msgstr "Erzwinge Großbuchstaben" #: cps/templates/config_edit.html:414 msgid "Enforce characters (needed For Chinese/Japanese/Korean Characters)" -msgstr "" +msgstr "Zeichen erzwingen (erforderlich für chinesische/japanische/koreanische Zeichen)" #: cps/templates/config_edit.html:418 msgid "Enforce special characters" @@ -2502,15 +2564,15 @@ msgstr "Anzahl in Übersicht anzuzeigender Autoren (0=alle werden angezeigt)" #: cps/templates/config_view_edit.html:40 cps/templates/readcbr.html:101 msgid "Theme" -msgstr "Theme" +msgstr "Design" #: cps/templates/config_view_edit.html:42 msgid "Standard Theme" -msgstr "Standard-Thema" +msgstr "Standard-Design" #: cps/templates/config_view_edit.html:43 msgid "caliBlur! Dark Theme" -msgstr "caliBlur! dunkles Thema" +msgstr "caliBlur! dunkles Design" #: cps/templates/config_view_edit.html:47 msgid "Regular Expression for Ignoring Columns" @@ -2589,80 +2651,65 @@ msgstr "Erlaubte/Verbotene Tags hinzufügen" msgid "Add Allowed/Denied custom column values" msgstr "Erlaubte/Verbotene Calibre Spalten hinzufügen" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Im Browser lesen" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Im Browser anhören" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Buch %(index)s von %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Herausgabedatum" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Als ungelesen markieren" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Als gelesen markieren" -#: cps/templates/detail.html:254 -#, fuzzy +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" -msgstr "Als ungelesen markieren" +msgstr "Buch als gelesen oder ungelesen markieren" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Gelesen" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Vom Archiv wiederherstellen" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Zum Archiv hinzufügen" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" -msgstr "" +msgstr "Buch als archiviert oder nicht markieren, um es in Calibre-Web auszublenden und vom Kobo Reader zu löschen" -#: cps/templates/detail.html:267 -#, fuzzy +#: cps/templates/detail.html:275 msgid "Archive" msgstr "Archiviert" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Beschreibung:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Zu Bücherregal hinzufügen" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Öffentlich)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Metadaten bearbeiten" #: cps/templates/email_edit.html:13 msgid "Email Account Type" -msgstr "Wähle Server Typ" +msgstr "Wähle Servertyp" #: cps/templates/email_edit.html:15 msgid "Standard Email Account" @@ -2726,10 +2773,6 @@ msgstr "Domainnamen eingeben" msgid "Denied Domains (Blacklist)" msgstr "Verbotene Domains für eine Registrierung" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Nächste" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Öffne die .kobo/Kobo/Kobo eReader.conf Datei in einem Texteditor und füge hinzu (oder ersetze):" @@ -2750,13 +2793,18 @@ msgstr "Calibre-Web Instanz ist nicht konfiguriert, bitte den Administrator kont msgid "Create Issue" msgstr "Issue erzeugen" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Datenbank-Konfiguration" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Zurück zur Hauptseite" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" -msgstr "Benutzer ausloggem" +msgstr "Benutzer ausloggen" #: cps/templates/index.html:71 msgid "Sort ascending according to download count" @@ -2818,19 +2866,19 @@ msgstr "Zufällige Bücher" #: cps/templates/index.xml:83 msgid "Books ordered by Author" -msgstr "Bücher nach Autoren sortiert" +msgstr "Bücher nach Autor sortiert" #: cps/templates/index.xml:92 msgid "Books ordered by publisher" -msgstr "Bücher nach Verlegern sortiert" +msgstr "Bücher nach Verleger sortiert" #: cps/templates/index.xml:101 msgid "Books ordered by category" -msgstr "Bücher nach Kategorien sortiert" +msgstr "Bücher nach Kategorie sortiert" #: cps/templates/index.xml:110 msgid "Books ordered by series" -msgstr "Bücher nach Serien sortiert" +msgstr "Bücher nach Serie sortiert" #: cps/templates/index.xml:119 msgid "Books ordered by Languages" @@ -2838,13 +2886,13 @@ msgstr "Bücher nach Sprache sortiert" #: cps/templates/index.xml:128 msgid "Books ordered by Rating" -msgstr "Bücher nach Bewertungen sortiert" +msgstr "Bücher nach Bewertung sortiert" #: cps/templates/index.xml:137 msgid "Books ordered by file formats" -msgstr "Bücher nach Dateiformaten sortiert" +msgstr "Bücher nach Dateiformat sortiert" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Bücherregale" @@ -2853,60 +2901,37 @@ msgstr "Bücherregale" msgid "Books organized in shelves" msgstr "Bücher in Bücherregalen organisiert" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Home" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Nagivation umschalten" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Bibiliothek durchsuchen" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Einfach" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Account" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Logout" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Lade hoch..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Fehler" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Hochladen beendet, verarbeite Daten, bitte warten..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Einstellungen" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Bitte die Seite nicht neu laden" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Durchsuchen" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Über" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Vorheriger Eintrag" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Buchdetails" @@ -2984,7 +3009,7 @@ msgstr "Dieses Buchformat wird permanent aus der Datenbank gelöscht" #: cps/templates/modal_dialogs.html:51 msgid "This book will be permanently erased from database" -msgstr "Das Buch wird aus der Calibre-Datenbank" +msgstr "Das Buch wird endgültig aus der Datenbank" #: cps/templates/modal_dialogs.html:52 msgid "and hard disk" @@ -2992,7 +3017,7 @@ msgstr "und von der Festplatte gelöscht" #: cps/templates/modal_dialogs.html:56 msgid "Important Kobo Note: deleted books will remain on any paired Kobo device." -msgstr "Wichtiger Kobo Hinweis: Gelöschte Bücher bleiben auf auf allen verbundenen Kobo Geräten erhalten." +msgstr "Wichtiger Kobo Hinweis: Gelöschte Bücher bleiben auf allen verbundenen Kobo Geräten erhalten." #: cps/templates/modal_dialogs.html:57 msgid "Books must first be archived and the device synced before a book can safely be deleted." @@ -3022,7 +3047,7 @@ msgstr "Übergeordnetes Verzeichnis" msgid "Select" msgstr "Auswahl" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3030,34 +3055,74 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web E-Book-Katalog" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" -msgstr "epub-Leser" +msgstr "EPUB-Leser" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +msgid "Choose a theme below:" +msgstr "Wähle ein Design:" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Hell" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Dunkel" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sepia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Schwarz" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Text umbrechen, wenn Seitenleiste geöffnet ist." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "Schriftgröße" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "Schriftart" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "Standard" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "Yahei" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "SimSun" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "KaiTi" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "Aufteilung" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "Zwei Spalten" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "Eine Spalte" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Comic-Leser" @@ -3234,10 +3299,6 @@ msgstr "Dieser Link wird in 10 Minuten ablaufen." msgid "Generate Series Cover Thumbnails" msgstr "Thumbnails für Serien Cover erzeugen" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Keine Ergebnisse" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Suchbegriff:" @@ -3254,13 +3315,13 @@ msgstr "Herausgabedatum von" msgid "Published Date To" msgstr "Herausgabedatum bis" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" -msgstr "" +msgstr "Beliebig" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" -msgstr "" +msgstr "Leer" #: cps/templates/search_form.html:60 msgid "Exclude Tags" @@ -3280,11 +3341,11 @@ msgstr "Sprachen ausschließen" #: cps/templates/search_form.html:127 msgid "Extensions" -msgstr "Datei Erweiterungen" +msgstr "Dateierweiterungen" #: cps/templates/search_form.html:135 msgid "Exclude Extensions" -msgstr "Datei Erweiterungen ausschliessen" +msgstr "Dateierweiterungen ausschliessen" #: cps/templates/search_form.html:145 msgid "Rating Above" @@ -3294,11 +3355,13 @@ msgstr "Bewertungen größer als" msgid "Rating Below" msgstr "Bewertungen kleiner als" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Von:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Bis:" @@ -3308,7 +3371,7 @@ msgstr "Lösche dieses Bücherregal" #: cps/templates/shelf.html:14 msgid "Edit Shelf Properties" -msgstr "Bücherregal Eigenschaften bearbeiten" +msgstr "Eigenschaften bearbeiten" #: cps/templates/shelf.html:17 msgid "Arrange books manually" @@ -3322,6 +3385,14 @@ msgstr "Manuelles Sortieren deaktivieren" msgid "Enable Change order" msgstr "Manuelles Sortieren aktivieren" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "Sortiere nach Buch zu Bücherregal hinzugefügt, neuestes zuerst" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "Sortiere nach Buch zu Bücherregal hinzugefügt, ältestes zuerst" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Bücherregal mit anderen Benutzern teilen" @@ -3332,7 +3403,7 @@ msgstr "Dieses Bücherregal mit Kobo synchronisieren" #: cps/templates/shelf_order.html:5 msgid "Drag to Rearrange Order" -msgstr "Drag 'n drop um Reihenfolge zu ändern" +msgstr "Ziehen und ablegen um Reihenfolge zu ändern" #: cps/templates/shelf_order.html:33 msgid "Hidden Book" @@ -3340,7 +3411,7 @@ msgstr "Verstecktes Buch" #: cps/templates/stats.html:7 msgid "Library Statistics" -msgstr "Bibiliotheksstatistiken" +msgstr "Bibliotheksstatistiken" #: cps/templates/stats.html:12 msgid "Books in this Library" @@ -3390,15 +3461,19 @@ msgstr "Fortschritt" msgid "Run Time" msgstr "Laufzeit" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "Meldung" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Aktionen" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Diese Aufgabe wird abgebrochen. Der Fortschritt wird gespeichert." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Dies ist eine geplante Aufgabe, sie wird zur geplanten Zeit erneut ausgeführt." @@ -3406,13 +3481,17 @@ msgstr "Dies ist eine geplante Aufgabe, sie wird zur geplanten Zeit erneut ausge msgid "Reset user Password" msgstr "Benutzerpasswort zurücksetzen" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "An die E-Mail-Adresse des E-Readers senden. Kommas verwenden, um E-Mails für mehrere E-Reader zu trennen" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Zeige nur Bücher mit dieser Sprache" #: cps/templates/user_edit.html:54 msgid "OAuth Settings" -msgstr "Oauth Einstellungen" +msgstr "OAuth Einstellungen" #: cps/templates/user_edit.html:56 msgid "Link" @@ -3432,7 +3511,7 @@ msgstr "Erzeugen/Ansehen" #: cps/templates/user_edit.html:70 msgid "Force full kobo sync" -msgstr "Komplettsynchronisation Kobo erzwingen" +msgstr "Kobo Komplettsynchronisation erzwingen" #: cps/templates/user_edit.html:88 msgid "Add allowed/Denied Custom Column Values" @@ -3516,7 +3595,7 @@ msgstr "Verbotene Spaltennamen" #: cps/templates/user_table.html:144 msgid "Change Password" -msgstr "Passworts ändern" +msgstr "Passwort ändern" #: cps/templates/user_table.html:147 msgid "View" @@ -3532,5 +3611,5 @@ msgstr "Ausgesuchte Bücherregale mit Kobo synchronisieren" #: cps/templates/user_table.html:156 msgid "Show Read/Unread Section" -msgstr "Zeige Gelesen/Ungelesen Auswahl" +msgstr "Zeige Gelesen/Ungelesen" diff --git a/cps/translations/el/LC_MESSAGES/messages.mo b/cps/translations/el/LC_MESSAGES/messages.mo index cb38d170..0f59d145 100644 Binary files a/cps/translations/el/LC_MESSAGES/messages.mo and b/cps/translations/el/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/el/LC_MESSAGES/messages.po b/cps/translations/el/LC_MESSAGES/messages.po index 05799f87..1a5fd286 100644 --- a/cps/translations/el/LC_MESSAGES/messages.po +++ b/cps/translations/el/LC_MESSAGES/messages.po @@ -6,7 +6,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" "Last-Translator: Depountis Georgios\n" "Language: el\n" @@ -15,512 +15,530 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Στατιστικά" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Ο διακομιστής επανεκκίνησε, παρακαλούμε φόρτωσε ξανά τη σελίδα" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Πραγματοποιείται κλείσιμο του διακομιστή, παρακαλούμε κλείσε το παράθυρο" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Άγνωστη εντολή" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Το βιβλίο έχει επιτυχώς μπει σε σειρά για αποστολή στο %(eReadermail)s" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "ʼΑγνωστο" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Σελίδα διαχειριστή" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Βασική Διαμόρφωση" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "UI Διαμόρφωση" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Η ειδικά προσαρμοσμένη στήλη No.%(column)d δεν υπάρχει στο επίπεδο βάσης δεδομένων" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Χρήστης Διαχειριστής" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Όλα" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Δεν βρέθηκε χρήστης" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Προβολή Όλων" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Δεν έχει απομείνει χρήστης διαχειριστής, δεν μπορεί να αφαιρεθεί ο ρόλος διαχειριστή" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Ενημερώθηκε η διαμόρφωση Calibre-Web" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Θέλεις πραγματικά να διαγράψεις τη Μονάδα Kobo;" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Είσαι σίγουρος/η πως θέλεις να διαγράψεις αυτό το ράφι;" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Είσαι σίγουρος/η πως θέλεις να διαγράψεις αυτό το ράφι;" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Είσαι σίγουρος/η πως θέλεις να διαγράψεις αυτό το ράφι;" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Είσαι σίγουρος/η πως θέλεις να διαγράψεις αυτό το ράφι;" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Είσαι σίγουρος/η πως θέλεις να κάνεις κλείσιμο;" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Απόρριψη" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Επιτρέπεται" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json Δεν Έχει Διαμορφωθεί Για Διαδικτυακή Εφαρμογή" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Το Φύλλο Καταγραφής Τοποθεσίας δεν είναι Έγκυρο, Παρακαλούμε Συμπλήρωσε Τη Σωστή Πορεία" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Η Πρόσβαση Φύλλου Καταγραφης Τοποθεσίας δεν είναι έγκυρη, Παρακαλούμε Συμπλήρωσε Τη Σωστή Πορεία" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Παρακαλούμε Συμπλήρωσε ένα Πάροχο LDAP, Θύρα, DN και Αντικείμενο Αναγνώρισης Χρήστη" -#: cps/admin.py:1211 +#: cps/admin.py:1223 #, fuzzy msgid "Please Enter a LDAP Service Account and Password" msgstr "Παρακαλούμε συμπλήρωσε ένα έγκυρο όνομα χρήστη για επαναφορά του κωδικού" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Το Αντικείμενο Φίλτρου Ομάδας LDAP Πρέπει να Έχει Μια \"%s\" Αναγνώριση Μορφής" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Το Αντικείμενο Φίλτρου Ομάδας LDAP Έχει Παρενθέσεις Που Δεν Ταιριάζουν" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Το Αντικείμενο Φίλτρου Χρήστη LDAP πρέπει να Έχει Μια \"%s\" Αναγνώριση Μορφής" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Το Αντικείμενο Φίλτρου Χρήστη LDAP Έχει Παρενθέσεις Που Δεν Ταιριάζουν" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Προσθήκη Νέου Χρήστη" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Επεξεργασία Ρυθμίσεων E-mail Διακομιστή" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Σφάλμα βάσης δεδομένων: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Παρουσιάστηκε σφάλμα κατά την αποστολή του δοκιμαστικού e-mail:% (res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Παρακαλούμε ρύθμισε πρώτα τη διεύθυνση e-mail σου..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Ενημερώθηκαν οι ρυθμίσεις E-mail διακομιστή" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Προέκυψε ένα άγνωστο σφάλμα. Παρακαλούμε δοκίμασε ξανά αργότερα." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Επεξεργασία χρήστη %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Κωδικός για επαναφορά %(user) χρήστη/ών" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Παρακαλούμε διαμόρφωσε πρώτα τις ρυθμίσεις ταχυδρομείου SMTP..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Προβολέας αρχείου φύλλου καταγραφής" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Αίτημα πακέτου ενημέρωσης" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Κατεβάζει πακέτο ενημέρωσης" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Ανοίγει πακέτο ενημέρωσης" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Αντικατάσταση αρχείων" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Οι συνδέσεις βάσης δεδομένων είναι κλειστές" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Σταματάει το διακομιστή" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Η ενημέρωση τελειώσε, παρακαλούμε πιέστε το εντάξει και φορτώστε ξανά τη σελίδα" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Η ενημέρωση απέτυχε:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP Σφάλμα" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Σφάλμα σύνδεσης" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Τελείωσε ο χρόνος κατά την προσπάθεια δημιουργίας σύνδεσης" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Γενικό σφάλμα" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Το Αρχείο Ενημέρωσης Δεν Μπόρεσε Να Αποθηκευτεί σε" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Αποτυχία Δημιουργίας Τουλάχιστον Ενός Χρήστη LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Αποτυχία Δημιουργίας Τουλάχιστον Ενός Χρήστη LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Σφάλμα: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Σφάλμα: Δεν επιστράφηκε χρήστης σε απάντηση του διακομιστή LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Τουλάχιστον Ένας Χρήστης LDAP Δεν Βρέθηκε Στη Βάση Δεδομένων" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Η Τοποθεσία DB δεν είναι Έγκυρη, Παρακαλούμε Συμπληρώστε Τη Σωστή Πορεία" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Η DB δεν μπορεί να Γραφτεί" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Το Αρχειο Κλειδί Τοποθεσίας δεν είναι Έγκυρο, Παρακαλούμε Συμπληρώστε Τη Σωστή Πορεία" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Η Τοποθεσία Certfile δεν είναι Έγκυρη, Παρακαλούμε Συμπληρώστε Τη Σωστή Πορεία" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Ενημερώθηκαν οι ρυθμίσεις E-mail διακομιστή" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Διαμόρφωση Λειτουργίας" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Παρακαλούμε συμπλήρωσε όλα τα πεδία!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Το E-mail δεν είναι από έγκυρο domain" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Προσθήκη νέου χρήστη" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Χρήστης/ες '%(user)s' δημιουργήθηκαν" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Βρέθηκε ένας ήδη υπάρχον λογαριασμός για αυτή τη διεύθυνση e-mail ή όνομα χρήστη." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Χρήστης/ες '%(nick)s' διαγράφηκαν" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Δεν έχει απομείνει χρήστης διαχειριστής, δεν μπορεί να διαγραφεί ο χρήστης" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Χρήστης/ες '%(nick)s' ενημερώθηκαν" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Αναζήτηση" + #: cps/converter.py:31 msgid "not installed" msgstr "δεν εγκαταστάθηκε" @@ -529,128 +547,123 @@ msgstr "δεν εγκαταστάθηκε" msgid "Execution permissions missing" msgstr "Λείπουν άδειες εκτέλεσης" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Η ειδικά προσαρμοσμένη στήλη No.%(column)d δεν υπάρχει στο επίπεδο βάσης δεδομένων" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Κανένα" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Oυπς! Ο επιλεγμένος τίτλος βιβλίου δεν είναι διαθέσιμος. Το αρχείο δεν υπάρχει ή δεν είναι προσβάσιμο" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Τα αναγνωριστικά δεν έχουν Διάκριση Πεζών-Κεφαλαίων Γραμμάτων, Αντικατάσταση Παλιού Αναγνωριστικού" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Τα μεταδεδομένα ενημερώθηκαν επιτυχώς" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Το αρχείο %(file)s ανέβηκε" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Η δομή πηγής ή προορισμού για μετατροπή λείπει" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Το βιβλίο είναι σε σειρά επιτυχώς για μετατροπή σε %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Υπήρξε ένα σφάλμα στη μετατροπή αυτού του βιβλίου: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Το βιβλίο που ανέβηκε πιθανόν να υπάρχει στη βιβλιοθήκη, σκέψου να το αλλάξεις πριν ανεβάσεις νέο: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Oυπς! Ο επιλεγμένος τίτλος βιβλίου δεν είναι διαθέσιμος. Το αρχείο δεν υπάρχει ή δεν είναι προσβάσιμο" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Τα αναγνωριστικά δεν έχουν Διάκριση Πεζών-Κεφαλαίων Γραμμάτων, Αντικατάσταση Παλιού Αναγνωριστικού" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s δεν είναι μια έγκυρη γλώσσα" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Τα μεταδεδομένα ενημερώθηκαν επιτυχώς" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Το βιβλίο που ανέβηκε πιθανόν να υπάρχει στη βιβλιοθήκη, σκέψου να το αλλάξεις πριν ανεβάσεις νέο: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Η επέκταση αρχείου '%(ext)s' δεν επιτρέπεται να ανέβει σε αυτό το διακομιστή" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Η επέκταση αρχείου '%(ext)s' δεν επιτρέπεται να ανέβει σε αυτό το διακομιστή" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Το αρχείο προς ανέβασμα πρέπει να έχει μια επέκταση" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Το αρχείο %(filename)s δεν μπόρεσε να αποθηκευτεί σε temp dir" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Αποτυχία Μετακίνησης Αρχείου Φόντου %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Η μορφή βιβλίου Διαγράφηκε Επιτυχώς" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Το Βιβλίο Διαγράφηκε Επιτυχώς" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "επεξεργασία μεταδεδομένων" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Αποτυχεία δημιουργίας πορείας %(path)s (Η άδεια απορρήφθηκε)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Αποτυχία αποθήκευσης αρχείου %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Μορφή αρχείου %(ext)s προστέθηκε σε %(book)s" @@ -663,486 +676,479 @@ msgstr "Η ρύθμιση του Google Drive δεν ολοκληρώθηκε, msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Η ανάκληση ονόματος δεν έχει επαληθευτεί, παρακαλούμε ακολούθησε τα βήματα για την επαλήθευση ονόματος στην κονσόλα προγραμματιστή google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s η δομή δεν βρέθηκε για την ταυτότητα βιβλίου: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s δεν βρέθηκε στο Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s δεν βρέθηκε: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Αποστολή στο Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Αυτό το e-mail έχει σταλεί μέσω Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web δοκιμαστικό e-mail" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Δοκιμαστικό e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Ξεκινήστε με το Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "e-mail εγγραφής για χρήστη: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Μετατροπή %(orig)s σε %(format)s και αποστολή στο Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Αποστολή %(format)s στο Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Αποστολή στο Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Το αρχείου που χητήθηκε δεν μπορεί να διαβαστεί. Μπορεί να υπάρχουν λαθασμένες άδειες;" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Η διαγραφή φακέλου βιβλίου για το βιβλίο %(id)s απέτυχε, η πορεία έχει υπό-φακέλους: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Η διαγραφή βιβλίου %(id)s απέτυχε: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, fuzzy, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Διαγραφή βιβλίου %(id)s, η πορεία βιβλίου δεν είναι έγκυρη: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Η μετονομασία τίτλου από: '%(src)s' σε '%(dest)s' απέτυχε με σφάλμα: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Το αρχείο %(file)s δεν βρέθηκε στο Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Η μετονομασία τίτλου από: '%(src)s' σε '%(dest)s' απέτυχε με σφάλμα: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Η πορεία βιβλίου %(path)s δεν βρέθηκε στο Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Αυτό το όνομα χρήστη έχει ήδη παρθεί" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Σφάλμα Κατεβάσματος Φόντου" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Σφάλμα Μορφής Φόντου" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Αποτυχία δημιουργίας πορείας για φόντο" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Το αρχείο φόντου δεν είναι ένα έγκυρο αρχείο εικόνας, ή δεν μπόρεσε να αποθηκευτεί" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Μόνο jpg/jpeg αρχεία υποστηρίζονται ως αρχεία φόντου" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Ανακάλυψε" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Δεν βρέθηκε δυαδικό αρχείο UnRar" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Σφάλμα εκτέλεσης UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "Η DB δεν μπορεί να Γραφτεί" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Λείπουν άδειες εκτέλεσης" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Σφάλμα εκτέλεσης UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Παρακαλούμε λάβε πρόσβαση στο calibre-web από ένα μη τοπικό εξηπηρετητή για να λάβεις μια έγκυρη api_endpoint για συσκευή kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Καθορισμός Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Εγγραφή με %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "τώρα έχεις συνδεθεί ως: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Η Σύνδεση στο %(oauth)s Πέτυχε" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Η σύνδεση απέτυχε, Δεν Υπάρχει Συνδεδεμένος Χρήστης Με Λογαριασμό OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Η αφαίρεση σύνδεσης με το %(oauth)s Πέτυχε" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Η αφαίρεση σύνδεσης με το %(oauth)s Απέτυχε" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Δεν Είναι Συνδεδεμένο με το %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Αποτυχία σύνδεσης με GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Αποτυχία συγκέντρωσης πληροφοριών χρήστη από το GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Αποτυχία σύνδεσης με το Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Αποτυχία συγκέντρωσης πληροφοριών χρήστη από το Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth σφάλμα, παρακαλούμε δοκίμασε ξανά αργότερα." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth σφάλμα, παρακαλούμε δοκίμασε ξανά αργότερα." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Σύνδεση" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Η μάρκα δεν βρέθηκε" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Η μάρκα έχει λήξει" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Επιτυχία! Παρακαλούμε επέστρεψε στη συσκευή σου" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Βιβλία" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Προβολή πρόσφατων βιβλίων" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Βιβλία στη Μόδα" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Προβολή Βιβλίων στη Μόδα" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Κατεβασμένα Βιβλία" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Προβολή Κατεβασμένων Βιβλίων" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Βιβλία με Κορυφαία Αξιολόγηση" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Προβολή Βιβλίων με Κορυφαία Αξιολόγηση" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Βιβλία που Διαβάστηκαν" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Προβολή διαβασμένων και αδιάβαστων" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Βιβλία που δεν Διαβάστηκαν" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Προβολή αδιάβαστων" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Ανακάλυψε" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Προβολή Τυχαίων Βιβλίων" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Κατηγορίες" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Προβολή επιλογών κατηγορίας" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Σειρές" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Προβολή επιλογών σειράς" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Συγγραφείς" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Προβολή επιλογών συγγραφέα" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Εκδότες" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Προβολή επιλογών εκδότη" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Γλώσσες" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Προβολή επιλογών γλώσσας" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Αξιολογήσεις" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Προβολή επιλογών αξιολόγησης" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Μορφές αρχείου" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Προβολή επιλογών μορφής αρχείου" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Αρχειοθετημένα Βιβλία" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Προβολή αρχειοθετημένων βιβλίων" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Λίστα Βιβλίων" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Προβολή Λίστας Βιβλίων" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Αναζήτηση" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Εκδόθηκε μετά" -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Εκδόθηκε πριν" -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Αξιολόγηση <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Αξιολόγηση >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Προχωρημένη Αναζήτηση" @@ -1198,7 +1204,7 @@ msgstr "Το βιβλίο έχει αφαιρεθεί από το ράφι: %(sn msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Δημιούργησε ένα Ράφι" @@ -1253,45 +1259,45 @@ msgstr "Ένα δημόσιο ράφι με το όνομα '%(title)s' υπάρ msgid "A private shelf with the name '%(title)s' already exists." msgstr "Ένα ιδιωτικό ράφι με το όνομα '%(title)s' υπάρχει ήδη." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Ράφι: '%(name)s" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Σφάλμα κατά το άνοιγμα του ραφιού. Το ράφι δεν υπάρχει ή δεν είναι προσβάσιμο" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Εργασίες" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Αναμονή" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Απέτυχε" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Ξεκίνησε" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Τελείωσε" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "ʼΑγνωστη κατάσταση" @@ -1324,178 +1330,178 @@ msgstr "Μια νέα ενημέρωση είναι διαθέσιμη. Κάνε msgid "No release information available" msgstr "Δεν υπάρχουν διαθέσιμες πληροφορίες αποδέσμευσης" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Ανακάλυψε (Τυχαία Βιβλία)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Βιβλία στη Μόδα (Με τα περισσότερα κατεβάσματα)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Κατεβασμένα βιβλία από %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Συγγραφέας: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Εκδότης: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Σειρές: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Αξιολόγηση: %(rating)s stars" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Μορφή αρχείου: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Κατηγορία: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Γλώσσα: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Κατεβασμένα" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Λίστα αξιολογήσεων" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Λίστα μορφών αρχείου" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Παρακαλούμε διαμόρφωσε πρώτα τις ρυθμίσεις ταχυδρομείου SMTP..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Το βιβλίο έχει επιτυχώς μπει σε σειρά για αποστολή στο %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Oυπς! Υπήρξε ένα σφάλμα κατά την αποστολή αυτού του βιβλίου: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Παρακαλούμε ενημέρωσε το προφίλ σου με μια έγκυρη Διεύθυνση E-mail Αποστολής στο Kindle." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Εγγραφή" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Ο διακομιστής E-Mail δεν έχει διαμορφωθεί, παρακαλούμε επικοινώνησε με το διαχειριστή σου!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Ο διακομιστής E-Mail δεν έχει διαμορφωθεί, παρακαλούμε επικοινώνησε με το διαχειριστή σου!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Η διεύθυνση e-mail σου δεν επιτρέπεται να εγγραφεί" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Το e-mail επιβεβαίωσης έχει σταλεί στον e-mail λογαριασμό σου." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Δεν μπόρεσε να ενεργοποιηθεί η επαλήθευση LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "τώρα έχεις συνδεθεί ως: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Εναλλακτική Σύνδεση ως: '%(nickname)s', Ο Διακομιστής LDAP δεν είναι προσβάσιμος, ή ο χρήστης δεν είναι γνωστός" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Δεν μπόρεσε να συνδεθεί: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Λανθασμένο Όνομα Χρήστη ή Κωδικός" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Ο Νέος Κωδικός έχει σταλεί στη διεύθυνση email σου" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Προέκυψε ένα άγνωστο σφάλμα. Παρακαλούμε δοκίμασε ξανά αργότερα." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Παρακαλούμε συμπλήρωσε ένα έγκυρο όνομα χρήστη για επαναφορά του κωδικού" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "τώρα έχεις συνδεθεί ως: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s's προφίλ" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Το προφίλ ενημερώθηκε" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Βρέθηκε ένας ήδη υπάρχον λογαριασμός για αυτή τη διεύθυνση e-mail." @@ -1504,54 +1510,58 @@ msgstr "Βρέθηκε ένας ήδη υπάρχον λογαριασμός γ msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Αποστολή στο Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Το εργαλείο μετατροπής Calibre ebook %(tool)s δεν βρέθηκε" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s μορφή δεν βρέθηκε σε δίσκο" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Ο μετατροπέας Ebook απέτυχε με άγνωστο σφάλμα" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Ο μετατροπέας Kepubify απέτυχε: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Το τροποποιημένο αρχείο δεν βρέθηκε ή υπάρχουν περισσότερα από ένα αρχεία στο φάκελο %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ο μετατροπέας Ebook απέτυχε: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Το Calibre απέτυχε με σφάλμα: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ο μετατροπέας Ebook απέτυχε: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1560,30 +1570,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "επεξεργασία μεταδεδομένων" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Ανέβασμα" @@ -1602,12 +1608,12 @@ msgstr "Όνομα Χρήστη" msgid "Email" msgstr "Διεύθυνση E-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Διεύθυνση E-mail Αποστολής στο Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Διαχειριστής" @@ -1617,8 +1623,8 @@ msgstr "Διαχειριστής" msgid "Password" msgstr "Κωδικός" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Κατέβασμα" @@ -1829,13 +1835,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Ακύρωση" @@ -1885,16 +1891,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "μείωση" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Περισσότερα από" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Γλώσσα" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Εκδότης" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Εκδόθηκε" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Περιγραφή" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Προηγούμενο" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Επόμενο" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Δεν Βρέθηκαν Αποτελέσματα" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Κεντρική" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Αναζήτηση Βιβλιοθήκης" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Αποσύνδεση" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Διαγραφή Βιβλίου" @@ -1923,99 +1989,107 @@ msgstr "Μετατροπή σε:" msgid "Convert book" msgstr "Μετατροπή βιβλίου" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Φόρτωση..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Κλείσιμο" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Σφάλμα" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Το ανέβασμα έγινε, γίνεται επεξεργασία, παρακαλούμε περίμενε..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Μορφή Ανεβάσματος" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Τίτλος Βιβλίου" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Συγγραφέας" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Ετικέτες" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Ταυτότητα Σειράς" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Ημερομηνία Έκδοσης" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Αξιολόγηση" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Περιγραφή" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Αναγνωριστικά" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Είδος Αναγνωριστικού" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Τιμή Αναγνωριστικού" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Αφαίρεση" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Προσθήκη Αναγνωριστικού" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Ετικέτες" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Ταυτότητα Σειράς" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Αξιολόγηση" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Συγκέντρωση Εξώφυλλου από URL (JPEG - Η εικόνα θα κατέβει και θα αποθηκευτεί σε βάση δεδομένων)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Ανέβασμα Εξώφυλλου από Τοπικό Δίσκο" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Ημερομηνία Έκδοσης" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Εκδότης" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Γλώσσα" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ναι" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Όχι" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Μορφή Ανεβάσματος" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Προβολή Βιβλίου σε Αποθήκευση" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Συγκέντρωση Μεταδεδομένων" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2023,38 +2097,32 @@ msgstr "Συγκέντρωση Μεταδεδομένων" msgid "Save" msgstr "Αποθήκευση" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Λέξη κλειδί" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr "Αναζήτηση λέξης κλειδιού" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Κάνε κλικ στο εξώφυλλο για φόρτωση μεταδεδομένων στη φόρμα" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Φόρτωση..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Κλείσιμο" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Πηγή" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Σφάλμα αναζήτησης!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Δεν βρέθηκε(αν) αποτέλεσμα(τα)! Παρακαλούμε δοκίμασε μια άλλη λέξη κλειδί." @@ -2163,7 +2231,7 @@ msgid "Enter " msgstr "Αναγνωριστικά" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Είσαι πραγματικά σίγουρος/η;" @@ -2642,74 +2710,61 @@ msgstr "Προσθήκη ετικετών Επιτρέπεται/Απορρίπ msgid "Add Allowed/Denied custom column values" msgstr "Προσθήκη τιμών ειδικά κατασκευασμένων στηλών Επιτρέπεται/Απορρίπτεται" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Διάβασε στο Πρόγραμμα Περιήγησης" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "ʼΑκουσε στο Πρόγραμμα Περιήγησης" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Εκδόθηκε" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Σήμανση ως Αδιάβαστο" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Σήμανση ως Διαβασμένο" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Σήμανση ως Αδιάβαστο" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Διαβάστηκε" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Επαναφορά από το αρχείο" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Προσθήκη στο αρχείο" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Αρχειοθετήθηκε" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Περιγραφή" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Προσθήκη στο ράφι" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Δημόσιο)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Επεξεργασία Μεταδεδομένων" @@ -2781,10 +2836,6 @@ msgstr "Όνομα domain" msgid "Denied Domains (Blacklist)" msgstr "Domains που Απορρίφθηκαν (Μαύρη λίστα)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Επόμενο" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "ʼΑνοιξε το .kobo/Kobo/Kobo eReader.conf αρχείο σε πρόγραμμα επεξεργασίας κειμένου και πρόσθεσε (ή κάνε επεξεργασία):" @@ -2807,11 +2858,16 @@ msgstr "Ο διακομιστής E-Mail δεν έχει διαμορφωθεί, msgid "Create Issue" msgstr "Δημιουργία Θέματος" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Διαμόρφωση Λειτουργίας" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Επιστροφή στην Κεντρική" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2901,7 +2957,7 @@ msgstr "Τα βιβλία ταξινομήθηκαν ανά Αξιολόγηση msgid "Books ordered by file formats" msgstr "Τα βιβλία ταξινομήθηκαν ανά μορφές αρχείου" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Ράφια" @@ -2910,60 +2966,37 @@ msgstr "Ράφια" msgid "Books organized in shelves" msgstr "Βιβλία οργανωμένα σε ράφια" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Κεντρική" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Αλλαγή Θέσης Περιήγησης" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Αναζήτηση Βιβλιοθήκης" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Απλό" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Λογαριασμός" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Αποσύνδεση" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Φόρτωση..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Σφάλμα" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Το ανέβασμα έγινε, γίνεται επεξεργασία, παρακαλούμε περίμενε..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Ρυθμίσεις" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Παρακαλούμε μην ανανεώσεις τη σελίδα" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Περιήγηση" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Σχετικά" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Προηγούμενο" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Λεπτομέρειες Βιβλίου" @@ -3079,7 +3112,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Βιβλίο" @@ -3088,36 +3121,81 @@ msgstr "Βιβλίο" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web Κατάλογος eBook" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "PDF πρόγραμμα ανάγνωσης" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Επιλογή ενός ονόματος χρήστη" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Φωτεινό" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Σκοτεινό" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Πίσω" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Επανάληψη ροής κειμένου όταν οι μπάρες στο πλάι είναι ανοιχτές." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Διαγραφή" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Αναμονή" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Κάθετο" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Διαβάστηκε" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3299,10 +3377,6 @@ msgstr "Αυτός ο σύνδεσμος επαλήθευσης θα λήξει msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Δεν Βρέθηκαν Αποτελέσματα" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Όρος Αναζήτησης:" @@ -3319,11 +3393,11 @@ msgstr "Ημερομηνία Έκδοσης Από" msgid "Published Date To" msgstr "Ημερομηνία Έκδοσης Μέχρι" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3360,11 +3434,13 @@ msgstr "Βαθμολογία Πάνω από" msgid "Rating Below" msgstr "Βαθμολογία Κάτω από" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3388,6 +3464,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Κοινοποίηση με Όλους" @@ -3456,15 +3540,20 @@ msgstr "Πρόοδος" msgid "Run Time" msgstr "Χρόνος Λειτουργίας" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Συγχώνευση" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3472,6 +3561,10 @@ msgstr "" msgid "Reset user Password" msgstr "Επαναφορά Κωδικού χρήστη" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Γλώσσα Βιβλίων" diff --git a/cps/translations/es/LC_MESSAGES/messages.mo b/cps/translations/es/LC_MESSAGES/messages.mo index 58e8775e..e1a55ea7 100644 Binary files a/cps/translations/es/LC_MESSAGES/messages.mo and b/cps/translations/es/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/es/LC_MESSAGES/messages.po b/cps/translations/es/LC_MESSAGES/messages.po index dc42003a..1930273d 100644 --- a/cps/translations/es/LC_MESSAGES/messages.po +++ b/cps/translations/es/LC_MESSAGES/messages.po @@ -5,525 +5,528 @@ # victorhck , 2018. # Angel Docampo , 2019. # Rafael A. Roa Hernández , 2020. +# adruki , 2024. msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" -"PO-Revision-Date: 2020-05-25 17:22+0200\n" -"Last-Translator: minakmostoles \n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-10-29 15:26+0100\n" +"Last-Translator: adruki \n" "Language: es\n" "Language-Team: es \n" "Plural-Forms: nplurals=2; plural=(n != 1);\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -# "Last-Translator: victorhck \n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Estadísticas" -#: cps/admin.py:150 -#, fuzzy +#: cps/admin.py:151 msgid "Server restarted, please reload page." -msgstr "Servidor reiniciado. Por favor, recargue la página" +msgstr "Servidor reiniciado. Por favor, recarga la página." -#: cps/admin.py:152 -#, fuzzy +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." -msgstr "El servidor se está apagando. Por favor, cierre la ventana" +msgstr "El servidor se está apagando. Por favor, cierra la ventana." -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" -msgstr "" +msgstr "Se ha reconectado la base de datos" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Comando desconocido" -#: cps/admin.py:174 -#, fuzzy +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" -msgstr "Puesto en cola un correo electrónico de prueba enviado a %(email)s, por favor, comprueba el resultado en Tareas" +msgstr "Libros en cola para la copia de seguridad de metadatos, por favor verifica las tareas para ver el resultado" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Desconocido" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Página de administración" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Configuración básica" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Configuración de la interfaz de usuario" -#: cps/admin.py:324 cps/templates/admin.html:51 -#, fuzzy +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "La columna personalizada n.º %(column)d no existe en la base de datos calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Editar usuarios" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Todo" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Usuario no encontrado" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} usuarios eliminados con éxito" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Mostrar todo" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" -msgstr "Petición mal formulada" +msgstr "Petición malformada" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "El nombre de invitado no se puede cambiar" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "El invitado no puede tener ese rol" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "No queda ningún usuario administrador, no se puede eliminar al usuario" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Valor tiene que ser verdadero o falso" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Rol inválido" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "El invitado no puede tener esta vista" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Vista no válida" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" -msgstr "El sitio del invitado se determina automáticamente y no puede ser cambiado" +msgstr "El idioma del invitado se determina automáticamente y no puede ser cambiado" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" -msgstr "No hay un sitio válido" +msgstr "No hay un idioma válido" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "No se ha indicado un idioma válido para el libro" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parámetro no encontrado" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Columna de lectura no válida" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Columna restringida no válida" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Configuración de Calibre-Web actualizada" -#: cps/admin.py:610 -msgid "Do you really want to delete the Kobo Token?" -msgstr "¿Realmente quieres borrar el Token de Kobo?" - -#: cps/admin.py:612 -msgid "Do you really want to delete this domain?" -msgstr "¿Realmente deseas borrar este dominio?" - -#: cps/admin.py:614 -msgid "Do you really want to delete this user?" -msgstr "¿Realmente quieres borrar este usuario?" - #: cps/admin.py:616 -msgid "Are you sure you want to delete this shelf?" -msgstr "¿Realmente quieres eliminar este estante?" +msgid "Do you really want to delete the Kobo Token?" +msgstr "¿Seguro que quieres eliminar el Token de Kobo?" #: cps/admin.py:618 -#, fuzzy -msgid "Are you sure you want to change locales of selected user(s)?" -msgstr "¿Realmente quieres cambiar el idioma de los usuarios seleccionados?" +msgid "Do you really want to delete this domain?" +msgstr "¿Seguro que deseas borrar este dominio?" #: cps/admin.py:620 -msgid "Are you sure you want to change visible book languages for selected user(s)?" -msgstr "¿Realmente quieres cambiar los idiomas visibles del libro de los usuarios seleccionados?" +msgid "Do you really want to delete this user?" +msgstr "¿Seguro que quieres borrar este usuario?" #: cps/admin.py:622 -msgid "Are you sure you want to change the selected role for the selected user(s)?" -msgstr "¿Realmente quieres cambiar el rol seleccionado de el usuario seleccionado?" +msgid "Are you sure you want to delete this shelf?" +msgstr "¿Estas seguro de que deseas eliminar este estante?" #: cps/admin.py:624 -#, fuzzy -msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" -msgstr "¿Realmente quieres cambiar las restricciones de los usuarios seleccionados?" +msgid "Are you sure you want to change locales of selected user(s)?" +msgstr "¿Estas seguro de que deseas quieres cambiar el idioma de los usuarios seleccionados?" #: cps/admin.py:626 +msgid "Are you sure you want to change visible book languages for selected user(s)?" +msgstr "¿Estas seguro de que deseas cambiar los idiomas visibles del libro de los usuarios seleccionados?" + +#: cps/admin.py:628 +msgid "Are you sure you want to change the selected role for the selected user(s)?" +msgstr "¿Estas seguro de que deseas cambiar el rol seleccionado de el usuario seleccionado?" + +#: cps/admin.py:630 +msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" +msgstr "¿Estas seguro de que deseas cambiar las restricciones de los usuarios seleccionados?" + +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" -msgstr "¿Realmente quieres cambiar las restricciones de visibilidad de los usuarios seleccionados?" +msgstr "¿Estas seguro de que deseas cambiar las restricciones de visibilidad de los usuarios seleccionados?" -#: cps/admin.py:629 -#, fuzzy +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" -msgstr "¿Realmente quieres cambiar el comportamiento de sincronización de estante del usuario seleccionado?" +msgstr "¿Estas seguro de que deseas cambiar el comportamiento de sincronización de estante del usuario seleccionado?" -#: cps/admin.py:631 -#, fuzzy +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" -msgstr "¿Realmente quieres cambiar la ubicación de la biblioteca Calibre?" +msgstr "¿Estas seguro de que deseas cambiar la ubicación de la biblioteca de Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" -msgstr "" +msgstr "Calibre-Web buscará portadas actualizadas y actualizará las miniaturas de las portadas, esto puede tardar un tiempo" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" -msgstr "" +msgstr "¿Estas seguro de que deseas eliminar la base de datos de sincronización de Calibre-Web para forzar una sincronización completa con tu lector Kobo?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" -msgstr "Denegar" +msgstr "Prohibir" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Permitir" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" -msgstr "" +msgstr "{} entradas de sincronización eliminadas" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Etiqueta no encontrada" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Acción no válida" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json no está configurado para la aplicación web" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "La ruta del Logfile no es válida. Por favor, Introduce la ruta correcta" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "La ruta del Access Logfile no es válida. Por favor, Introduce la ruta correcta" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Por favor, Introduce un proveedor LDAP, puerto, DN y el User Object Identifier" -#: cps/admin.py:1211 -#, fuzzy +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Por favor, introduce una cuenta de servicio LDAP y su contraseña" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Por favor, introduce una cuenta de servicio LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP Group Object Filter necesita tener un identificador de formato \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "El LDAP Group Object Filter tiene un paréntesis diferente" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP Group Object Filter necesita tener un identificador de formato \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "El LDAP Group Object Filter tiene un paréntesis diferente" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "El filtro de usuarios LDAP necesita tener un identificador de formato \"%s\"" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "El filtro de LDAP \"Member User\" tiene paréntesis no coincidentes" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "Ubicaciones del certificado de la CA del LDAP, del certificado o de la clave no válidos. Por favor introduce la ruta correcta" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Añadir nuevo usuario" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Cambiar parámetros de correo" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." -msgstr "" +msgstr "Cuenta de Gmail verificada." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Error en la base de datos: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Puesto en cola un correo electrónico de prueba enviado a %(email)s, por favor, comprueba el resultado en Tareas" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Ocurrió un error enviando el correo electrónico de prueba: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." -msgstr "Por favor, configure su correo electrónico primero..." +msgstr "Por favor, configura tu correo electrónico primero..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Actualizados los ajustes del servidor de correo electrónico" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" -msgstr "" +msgstr "Editar la configuración de tareas programadas" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" -msgstr "" +msgstr "Hora de inicio no válida para la tarea especificada" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" -msgstr "" +msgstr "Duración no válida para la tarea especificada" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" -msgstr "" +msgstr "Configuración de tareas programadas actualizada" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Ha ocurrido un error desconocido. Por favor vuelva a intentarlo más tarde." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" -msgstr "" +msgstr "La base de datos de configuraciones no es modificable" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" -msgstr "Editar Usuario %(nick)s" +msgstr "Editar usuario %(nick)s" -#: cps/admin.py:1445 -#, fuzzy, python-format +#: cps/admin.py:1457 +#, python-format msgid "Success! Password for user %(user)s reset" -msgstr "Contraseña para el usuario %(user)s reinicializada" +msgstr "Restablecida contraseña para el usuario %(user)s" -#: cps/admin.py:1451 -#, fuzzy +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." -msgstr "Configura primero los parámetros del servidor SMTP..." +msgstr "Por favor, configura los ajustes de correo SMTP." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Visor del fichero de log" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Solicitando paquete de actualización" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Descargando paquete de actualización" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Descomprimiendo paquete de actualización" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Remplazando archivos" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Los conexiones con la base datos están cerradas" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Parando el servidor" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Actualización finalizada. Por favor, pulse OK y recargue la página" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Falló la actualización:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Error HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Error de conexión" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Tiempo agotado mientras se trataba de establecer la conexión" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Error general" -#: cps/admin.py:1539 -#, fuzzy +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" -msgstr "La actualización del archivo no pudo guardarse en el directorio temporal (Temp Dir)" +msgstr "La actualización del archivo no pudo guardarse en el directorio temporal" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" -msgstr "" +msgstr "Los archivos no se pudieron reemplazar durante la actualización" -#: cps/admin.py:1564 -#, fuzzy +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" -msgstr "Error al crear al menos un usuario LDAP" +msgstr "Error al extraer al menos un usuario LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Error al crear al menos un usuario LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Error: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Error: el servidor LDAP no ha devuelto ningún usuario" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Al menos, un usuario LDAP no se ha encontrado en la base de datos" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Usuario importado con éxito" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "La ruta de los libros no es válida" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "La ruta de la base de datos no es válida. Por favor, Introduce la ruta correcta" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "La base de datos no es modificable" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "La ruta del Keyfile no es válida, por favor, Introduce la ruta correcta" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "La ruta de Certfile no es válida, por favor, Introduce la ruta correcta" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" -msgstr "" +msgstr "La longitud de la contraseña debe estar entre 1 y 40 caracteres" -#: cps/admin.py:1894 -#, fuzzy +#: cps/admin.py:1917 msgid "Database Settings updated" -msgstr "Actualizados los ajustes del servidor de correo electrónico" +msgstr "Actualizados los ajustes de la base de datos" -#: cps/admin.py:1902 -#, fuzzy +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Configuración de la base de datos" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." -msgstr "¡Por favor, rellena todos los campos!" +msgstr "Por favor, rellena todos los campos." -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "El correo electrónico no tiene un dominio válido" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Añadir un nuevo usuario" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Usuario '%(user)s' creado" -#: cps/admin.py:1949 -#, fuzzy +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." -msgstr "Encontrada una cuenta existente para este correo electrónico o nombre de usuario." +msgstr "Ya existe una cuenta para este correo electrónico o nombre." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Usuario '%(nick)s' eliminado" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "No puedes borrar al Usuario Invitado" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "No queda ningún usuario administrador, no se puede borrar al usuario" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" -msgstr "" +msgstr "El correo electrónico no puede estar vacío y debe ser un correo electrónico válido" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" -msgstr "Usuario '%(nick)s' actualizado" +msgstr "Usuario '%(nick)s' actualizado" + +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Buscar" #: cps/converter.py:31 msgid "not installed" @@ -533,128 +536,122 @@ msgstr "no instalado" msgid "Execution permissions missing" msgstr "Faltan permisos de ejecución" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Columna personalizada No.%(column)d no existe en la base de datos calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" -msgstr "Ninguno" +msgstr "Ninguna" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "oh, oh, el libro seleccionado no está disponible. El archivo no existe o no es accesible" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Los identificadores no distinguen entre mayúsculas y minúsculas, sobrescribiendo el identificador antiguo" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadatos actualizados con éxito" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "El fichero %(file)s ha sido subido" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Falta la fuente o el formato de destino para la conversión" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Libro puesto a la cola para su conversión a %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Ocurrió un error al convertir este libro: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "El libro cargado probablemente existe en la biblioteca, considera cambiarlo antes de subirlo de nuevo: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "El libro seleccionado no está disponible. El archivo no existe o no es accesible" -#: cps/editbooks.py:718 cps/editbooks.py:1049 -#, fuzzy, python-format +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "El usuario no tiene derechos para subir la portada" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Los identificadores no distinguen entre mayúsculas y minúsculas, sobrescribiendo el identificador antiguo" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 +#, python-format msgid "'%(langname)s' is not a valid language" -msgstr "%(langname)s no es un idioma válido" +msgstr "'%(langname)s' no es un idioma válido" -#: cps/editbooks.py:756 cps/editbooks.py:1192 -#, fuzzy +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadatos actualizados con éxito" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Error al editar el libro: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "El libro subido probablemente ya existe en la biblioteca, considera cambiarlo antes de subir uno nuevo: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" -msgstr "No se permite subir archivos con la extensión '%(ext)s' a este servidor" +msgstr "No se permite subir este tipo de archivo a este servidor" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "No se permite subir archivos con la extensión '%(ext)s' a este servidor" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "El archivo a subir debe tener una extensión" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "El archivo %(filename)s no pudo salvarse en el directorio temporal (Temp Dir)" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" -msgstr "Fallo al mover el archivo de cubierta %(file)s: %(error)s" +msgstr "Fallo al mover el archivo de portada %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Formato de libro eliminado con éxito" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Libro eliminado con éxito" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" -msgstr "" +msgstr "Te faltan permisos para eliminar libros" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "editar metadatos" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" -msgstr "%(seriesindex) no es un número válido, saltando" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "Seriesindex: %(seriesindex)s no es un número válido, saltando" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" -msgstr "" +msgstr "El usuario no tiene derechos para subir formatos de archivo adicionales" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." -msgstr "Fallo al crear la ruta %(path)s (permiso denegado)" +msgstr "Error al crear la ruta %(path)s (permiso denegado)" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." -msgstr "Fallo al guardar el archivo %(file)s." +msgstr "Error al guardar el archivo %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Archivo con formato %(ext)s añadido a %(book)s" @@ -665,489 +662,463 @@ msgstr "La configuración de Google Drive no se ha completado, intente desactiva #: cps/gdrive.py:96 msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" -msgstr "El dominio Callback no se ha verificado, siga los pasos para verificarlo en la consola de desarrollador de Google" +msgstr "El dominio de retorno (callback) no está verificado, por favor sigue los pasos para verificar el dominio en la consola de desarrolladores de Google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s formato no encontrado para el id del libro: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s no encontrado en Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s no encontrado: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 -#, fuzzy +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" -msgstr "Enviar al Kindle" +msgstr "Enviar al eReader" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 -#, fuzzy +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "Este correo electrónico ha sido enviado usando Calibre-Web." -#: cps/helper.py:120 -#, fuzzy +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "Correo de prueba de Calibre-Web" -#: cps/helper.py:121 -#, fuzzy +#: cps/helper.py:124 msgid "Test Email" msgstr "Comprobar correo electrónico" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Primeros pasos con Calibre-Web" -#: cps/helper.py:143 -#, fuzzy, python-format +#: cps/helper.py:146 +#, python-format msgid "Registration Email for user: %(name)s" msgstr "Correo electrónico de registro para el usuario: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 -#, fuzzy, python-format +#: cps/helper.py:157 cps/helper.py:163 +#, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" -msgstr "Convertir %(orig)s a %(format)s y enviar al Kindle" +msgstr "Convertir %(orig)s a %(format)s y enviar al eReader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 -#, fuzzy, python-format +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 +#, python-format msgid "Send %(format)s to eReader" -msgstr "Enviado %(format)s al Kindle" +msgstr "Enviar %(format)s al eReader" -#: cps/helper.py:227 -#, fuzzy, python-format +#: cps/helper.py:230 +#, python-format msgid "%(book)s send to eReader" -msgstr "Enviar al Kindle" +msgstr "%(book)s enviado al eReader" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "El archivo solicitado no puede ser leído. ¿Quizás existen problemas con los permisos?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" -msgstr "" +msgstr "No se pudo establecer el estado de lectura: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Fallo al intentar borrar la carpeta del libro %(id)s, la ruta tiene subcarpetas: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" -msgstr "El eliminado del libro %(id)s falló: %(message)s" +msgstr "La eliminación del libro %(id)s falló: %(message)s" -#: cps/helper.py:387 -#, fuzzy, python-format +#: cps/helper.py:392 +#, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" -msgstr "Borrando el libro %(id)s, la ruta del libro es no válida: %(path)s" +msgstr "Eliminando el libro %(id)s solo de la base de datos, la ruta del libro en la base de datos no es válida: %(path)s" -#: cps/helper.py:463 -#, fuzzy, python-format +#: cps/helper.py:439 +#, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "El renombrado del título de: '%(src)s' a '%(dest)s' falló con el error: %(error)s" +msgstr "El cambio de nombre del autor de '%(src)s' a '%(dest)s' falló con el error: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" -msgstr "Fichero %(file)s no encontrado en Google Drive" +msgstr "Archivo %(file)s no encontrado en Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "El renombrado del título de: '%(src)s' a '%(dest)s' falló con el error: %(error)s" +msgstr "El cambio de nombre del título de '%(src)s' a '%(dest)s' falló con el error: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "La ruta %(path)s del libro no fue encontrada en Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" -msgstr "" +msgstr "Se encontró una cuenta existente para esta dirección de correo electrónico" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Este nombre de usuario ya está en uso" -#: cps/helper.py:702 -#, fuzzy +#: cps/helper.py:679 msgid "Invalid Email address format" -msgstr "Dirección de correo no válida" +msgstr "Formato de dirección de correo electrónico no válido" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" -msgstr "" +msgstr "La contraseña no cumple con las reglas de validación de contraseñas" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" -msgstr "" +msgstr "El módulo de Python 'advocate' no está instalado, pero se necesita para subir portadas" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" -msgstr "Error al descargar la cubierta" +msgstr "Error al descargar la portada" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" -msgstr "Error en el formato de la cubierta" +msgstr "Error en el formato de la portada" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" -msgstr "" +msgstr "No tienes permiso para acceder a localhost o a la red local para subir portadas" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" -msgstr "Error al crear una ruta para la cubierta" +msgstr "Error al crear una ruta para la portada" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" -msgstr "El archivo de cubierta no es una imágen válida" +msgstr "El archivo de la portada no es una imagen válida o no se pudo almacenar" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Sólo se admiten como portada los archivos jpg/jpeg/png/webp/bmp" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" -msgstr "" +msgstr "Contenido del archivo de la portada no válido" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Sólo se admiten como portada los archivos jpg/jpeg" -#: cps/helper.py:1011 cps/helper.py:1168 -#, fuzzy +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" -msgstr "Descubrir" +msgstr "Portada" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" -msgstr "No se encuentra el archivo binario UnRar" +msgstr "No se encontró el archivo binario de UnRar." -#: cps/helper.py:1039 -#, fuzzy +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "Error ejecutando UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" -msgstr "" +msgstr "No se pudo encontrar el directorio especificado" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" -msgstr "" +msgstr "Por favor, especifica un directorio, no un archivo" -#: cps/helper.py:1064 -#, fuzzy +#: cps/helper.py:1042 msgid "Calibre binaries not viable" -msgstr "La base de datos no es modificable" +msgstr "Los binarios de Calibre no son viables" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" -msgstr "" +msgstr "Faltan los binarios de Calibre: %(missing)s" -#: cps/helper.py:1075 -#, fuzzy, python-format +#: cps/helper.py:1053 +#, python-format msgid "Missing executable permissions: %(missing)s" -msgstr "Faltan permisos de ejecución" +msgstr "Faltan permisos de ejecución: %(missing)s" -#: cps/helper.py:1080 -#, fuzzy +#: cps/helper.py:1058 msgid "Error executing Calibre" -msgstr "Error ejecutando UnRar" +msgstr "Error ejecutando Calibre" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" -msgstr "" +msgstr "Poner en cola todos los libros para la copia de seguridad de metadatos" -#: cps/kobo_auth.py:90 -#, fuzzy +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" -msgstr "Por favor, accede a calibre-web desde una ubicación que no sea localhost para obtener una api_endpoint válida para tu dispositivo Kobo" +msgstr "Por favor, accede a Calibre-Web desde una dirección que no sea localhost para obtener un endpoint de API válido para el dispositivo Kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Configuración de Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Registrado con %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" -msgstr "has iniciado sesión como : '%(nickname)s'" +msgstr "Has iniciado sesión como : '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "El enlace a %(oauth)s se ha realizado con éxito" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" -msgstr "Acceso erróneo, ningún usuario enlazado con la cuenta OAuth" +msgstr "Inicio de sesión fallido, no hay ningún usuario vinculado con la cuenta de OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" -msgstr "%(oauth)s desenlazado con éxito" +msgstr "Desvinculación de %(oauth)s completada con éxito" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" -msgstr "Error al desenlazar %(oauth)s" +msgstr "Fallo al desvincular %(oauth)s" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "No vinculado con %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Error al iniciar sesión con GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Error al obtener la información del usuario de GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Error al iniciar sesión con Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Error al obtener información del usuario de Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." -msgstr "Error en GitHub Oauth, por favor, vuelva a intentarlo más tarde." +msgstr "Error en GitHub Oauth, por favor, vuelve a intentarlo más tarde." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Error GitHub Oauth {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." -msgstr "Error en Google Oauth, por favor vuelva a intentarlo más tarde." +msgstr "Error en Google Oauth, por favor vuelve a intentarlo más tarde." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Error Google Oauth {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" -msgstr "{} Estrellas" +msgstr "{} estrellas" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Inicio de sesión" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token no encontrado" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "El token ha expirado" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" -msgstr "¡Correcto! Por favor regrese a su dispositivo" +msgstr "¡Hecho! Por favor regresa a tu dispositivo" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Libros" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Mostrar libros recientes" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Libros populares" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Mostrar libros populares" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" -msgstr "Libros Descargados" +msgstr "Libros descargados" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" -msgstr "Mostrar Libros Descargados" +msgstr "Mostrar libros descargados" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Libros mejor valorados" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Mostrar libros mejor valorados" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Libros leídos" -#: cps/render_template.py:64 -#, fuzzy +#: cps/render_template.py:63 msgid "Show Read and Unread" msgstr "Mostrar leídos y no leídos" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Libros no leídos" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Mostrar no leído" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Descubrir" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Mostrar libros al azar" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Categorías" -#: cps/render_template.py:74 cps/templates/user_table.html:158 -#, fuzzy +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" -msgstr "Mostrar selección de categorías" +msgstr "Mostrar sección de categorías" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Series" -#: cps/render_template.py:77 cps/templates/user_table.html:157 -#, fuzzy +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" -msgstr "Mostrar selección de series" +msgstr "Mostrar sección de series" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autores" -#: cps/render_template.py:80 cps/templates/user_table.html:160 -#, fuzzy +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" -msgstr "Mostrar selección de autores" +msgstr "Mostrar sección de autores" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Editores" -#: cps/render_template.py:84 cps/templates/user_table.html:163 -#, fuzzy +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" -msgstr "Mostrar selección de editores" +msgstr "Mostrar sección de editores" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Idiomas" -#: cps/render_template.py:88 cps/templates/user_table.html:155 -#, fuzzy +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" -msgstr "Mostrar selección de idiomas" +msgstr "Mostrar sección de idiomas" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" -msgstr "Calificaciones" +msgstr "Valoraciones" -#: cps/render_template.py:91 cps/templates/user_table.html:164 -#, fuzzy +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" -msgstr "Mostrar selección de calificaciones" +msgstr "Mostrar sección de valoraciones" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formatos de archivo" -#: cps/render_template.py:94 cps/templates/user_table.html:165 -#, fuzzy +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" -msgstr "Mostrar selección de formatos de archivo" +msgstr "Mostrar sección de formatos de archivo" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Libros archivados" -#: cps/render_template.py:98 cps/templates/user_table.html:166 -#, fuzzy +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "Mostrar libros archivados" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" -msgstr "Lista de Libros" +msgstr "Lista de libros" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" -msgstr "Mostrar Lista de Libros" +msgstr "Mostrar lista de libros" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Buscar" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Publicado después de " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Publicado antes de " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" -msgstr "Calificación <= %(rating)s" +msgstr "Valoración <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" -msgstr "Calificación >= %(rating)s" +msgstr "Valoración >= %(rating)s" -#: cps/search.py:221 -#, fuzzy, python-format +#: cps/search.py:234 +#, python-format msgid "Read Status = '%(status)s'" -msgstr "Estado de lectura = $(status)s" +msgstr "Estado de lectura = '%(status)s'" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Error en la búsqueda de columnas personalizadas, por favor reinicia Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Búsqueda avanzada" @@ -1156,93 +1127,90 @@ msgid "Invalid shelf specified" msgstr "Estantería especificada no válida" #: cps/shelf.py:55 -#, fuzzy msgid "Sorry you are not allowed to add a book to that shelf" -msgstr "Lo sentimos, no tiene permisos para agregar un libro al estante: %(shelfname)s" +msgstr "Lo siento, no tienes permiso para añadir un libro a esa estantería" #: cps/shelf.py:64 #, python-format msgid "Book is already part of the shelf: %(shelfname)s" -msgstr "El libro ya forma parte del estante: %(shelfname)s" +msgstr "El libro ya forma parte de la estantería %(shelfname)s" #: cps/shelf.py:77 #, python-format msgid "%(book_id)s is a invalid Book Id. Could not be added to Shelf" -msgstr "" +msgstr "%(book_id)s es un ID de libro no válido. No se pudo añadir la estantería" #: cps/shelf.py:97 #, python-format msgid "Book has been added to shelf: %(sname)s" -msgstr "El libro fue agregado a el estante: %(sname)s" +msgstr "El libro fue agregado a la estantería: %(sname)s" #: cps/shelf.py:116 msgid "You are not allowed to add a book to the shelf" -msgstr "" +msgstr "Lo siento, no tienes permiso para añadir un libro a la estantería" #: cps/shelf.py:134 #, python-format msgid "Books are already part of the shelf: %(name)s" -msgstr "Los libros ya forman parte del estante: %(name)s" +msgstr "Los libros ya forman parte de la estantería: %(name)s" #: cps/shelf.py:146 #, python-format msgid "Books have been added to shelf: %(sname)s" -msgstr "Los libros han sido añadidos al estante: %(sname)s" +msgstr "Los libros han sido añadidos a la estantería: %(sname)s" #: cps/shelf.py:153 #, python-format msgid "Could not add books to shelf: %(sname)s" -msgstr "No se pudieron agregar libros al estante: %(sname)s" +msgstr "No se pudieron agregar libros a la estantería: %(sname)s" #: cps/shelf.py:199 #, python-format msgid "Book has been removed from shelf: %(sname)s" -msgstr "El libro fue eliminado del estante: %(sname)s" +msgstr "El libro fue eliminado de la estantería: %(sname)s" #: cps/shelf.py:208 msgid "Sorry you are not allowed to remove a book from this shelf" -msgstr "" +msgstr "Lo siento, no tienes permiso para eliminar un libro de esta estantería" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" -msgstr "Crear un estante" +msgstr "Crear una estantería" #: cps/shelf.py:226 -#, fuzzy msgid "Sorry you are not allowed to edit this shelf" -msgstr "Lo siento, no tiene permiso para eliminar un libro del estante: %(sname)s" +msgstr "Lo siento, no tienes permiso para editar este estante" #: cps/shelf.py:228 msgid "Edit a shelf" -msgstr "Editar un estante" +msgstr "Editar una estantería" #: cps/shelf.py:237 msgid "Error deleting Shelf" -msgstr "" +msgstr "Error al eliminar la estantería" #: cps/shelf.py:239 -#, fuzzy msgid "Shelf successfully deleted" -msgstr "Libro eliminado con éxito" +msgstr "Estantería eliminada correctamente" #: cps/shelf.py:289 #, python-format msgid "Change order of Shelf: '%(name)s'" -msgstr "Cambiar orden del estante: '%(name)s'" +msgstr "Cambiar orden de la estantería: '%(name)s'" #: cps/shelf.py:324 msgid "Sorry you are not allowed to create a public shelf" -msgstr "" +msgstr "Lo siento, no tienes permiso para crear una estantería pública" #: cps/shelf.py:341 #, python-format msgid "Shelf %(title)s created" -msgstr "Estante %(title)s creado" +msgstr "Estantería %(title)s creada" #: cps/shelf.py:344 #, python-format msgid "Shelf %(title)s changed" -msgstr "Estante %(title)s cambiado" +msgstr "Estantería %(title)s cambiada" #: cps/shelf.py:358 msgid "There was an error" @@ -1251,58 +1219,58 @@ msgstr "Ha sucedido un error" #: cps/shelf.py:380 #, python-format msgid "A public shelf with the name '%(title)s' already exists." -msgstr "Ya existe un estante público con el nombre '%(title)s'." +msgstr "Ya existe una estantería pública con el nombre '%(title)s'." #: cps/shelf.py:391 #, python-format msgid "A private shelf with the name '%(title)s' already exists." -msgstr "Ya existe un estante privado con el nombre '%(title)s'." +msgstr "Ya existe una estantería privada con el nombre '%(title)s'." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" -msgstr "Estante: '%(name)s'" +msgstr "Estantería: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" -msgstr "Error al abrir un estante. El estante no existe o no es accesible" +msgstr "Error al abrir una estantería. La estantería no existe o no es accesible" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tareas" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Esperando" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Fallido" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" -msgstr "Comenzado" +msgstr "Iniciado" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" +msgstr "Terminado" + +#: cps/tasks_status.py:71 +msgid "Ended" msgstr "Finalizado" -#: cps/tasks_status.py:70 -msgid "Ended" -msgstr "" - -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" -msgstr "" +msgstr "Cancelado" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Estado desconocido" #: cps/updater.py:433 cps/updater.py:444 cps/updater.py:545 cps/updater.py:560 msgid "Unexpected data while reading update information" -msgstr "Dato inesperado mientras se leía la información de actualización" +msgstr "Datos inesperados al leer la información de actualización" #: cps/updater.py:440 cps/updater.py:552 msgid "No update available. You already have the latest version installed" @@ -1310,7 +1278,7 @@ msgstr "Actualización no disponible. Ya tienes instalada la versión más recie #: cps/updater.py:458 msgid "A new update is available. Click on the button below to update to the latest version." -msgstr "Una nueva actualización está disponible. Haz clic en el botón inferior para actualizar a la versión más reciente." +msgstr "Hay una nueva actualización disponible. Haz clic en el botón de abajo para actualizar a la versión más reciente." #: cps/updater.py:476 msgid "Could not fetch update information" @@ -1327,268 +1295,257 @@ msgstr "Hay una nueva actualización disponible. Haz clic en el botón de abajo #: cps/updater.py:538 msgid "No release information available" -msgstr "No hay información del lanzamiento disponible" +msgstr "No hay información de lanzamiento disponible" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Descubrir (Libros al azar)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Libros populares (los más descargados)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Libros descargados por %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Autor/es: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Editor/es: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Series: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" -msgstr "" +msgstr "Valoración: Ninguna estrella" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" -msgstr "Calificación: %(rating)s estrellas" +msgstr "Valoración: %(rating)s estrellas" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Formato del archivo: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Categoría : %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Idioma: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Descargas" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" -msgstr "Lista de calificaciones" +msgstr "Lista de valoraciones" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Lista de formatos" -#: cps/web.py:1249 -#, fuzzy +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." -msgstr "Configura primero los parámetros del servidor SMTP..." +msgstr "Por favor, configura primero los ajustes de correo SMTP..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Libro puesto en la cola de envío a %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" -msgstr "Ha sucedido un error en el envío del libro: %(res)s" +msgstr "Hubo un error en el envío del libro: %(res)s" -#: cps/web.py:1261 -#, fuzzy +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." -msgstr "Por favor actualiza tu perfil con la dirección de correo de su kindle..." +msgstr "Por favor, actualiza tu perfil con un correo electrónico de eReader válido." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" -msgstr "" +msgstr "Por favor, espera un minuto para registrar el siguiente usuario" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registro" -#: cps/web.py:1281 cps/web.py:1385 -#, fuzzy +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" -msgstr "El servidor de correo no está configurado, por favor, ¡avisa a tu administrador!" +msgstr "Error de conexión con el backend de Limiter, por favor contacta con tu administrador" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." -msgstr "El servidor de correo no está configurado, por favor, ¡avisa a tu administrador!" +msgstr "El servidor de correo no está configurado, por favor contacta con tu administrador." -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." -msgstr "Su correo electrónico no está permitido para registrarse" +msgstr "Su correo electrónico no está permitido para registrarse." -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Se ha enviado un correo electrónico de verificación a su cuenta de correo." -#: cps/web.py:1368 cps/web.py:1391 -#, fuzzy +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "No se puede activar la autenticación LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" -msgstr "" +msgstr "Por favor, espera un minuto antes de iniciar sesión de nuevo" -#: cps/web.py:1400 -#, fuzzy, python-format +#: cps/web.py:1408 +#, python-format msgid "you are now logged in as: '%(nickname)s'" -msgstr "has iniciado sesión como : '%(nickname)s'" +msgstr "estás ahora conectado como: '%(nickname)s'" -#: cps/web.py:1407 -#, fuzzy, python-format +#: cps/web.py:1415 +#, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" -msgstr "Fallback login como: '%(nickname)s', no se puede acceder al servidor LDAP o usuario desconocido" +msgstr "Inicio de sesión alternativo como: '%(nickname)s', servidor LDAP no accesible o usuario no conocido" -#: cps/web.py:1412 -#, fuzzy, python-format +#: cps/web.py:1420 +#, python-format msgid "Could not login: %(message)s" -msgstr "No se pudo entrar: %(message)s" +msgstr "No se pudo iniciar sesión: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 -#, fuzzy +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" -msgstr "Usuario o contraseña inválido" +msgstr "Nombre de usuario o contraseña incorrectos" -#: cps/web.py:1423 -#, fuzzy +#: cps/web.py:1431 msgid "New Password was sent to your email address" -msgstr "Una nueva contraseña se ha enviado a su cuenta de correo electrónico" +msgstr "Se envió una nueva contraseña a tu dirección de correo electrónico" -#: cps/web.py:1427 -#, fuzzy +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "Ha ocurrido un error desconocido. Por favor vuelva a intentarlo más tarde." -#: cps/web.py:1429 -#, fuzzy -msgid "Please enter valid username to reset password" -msgstr "Por favor, introduce un usuario válido para restablecer la contraseña" - #: cps/web.py:1437 -#, fuzzy, python-format -msgid "You are now logged in as: '%(nickname)s'" -msgstr "has iniciado sesión como : '%(nickname)s'" +msgid "Please enter valid username to reset password" +msgstr "Por favor, introduce un nombre de usuario válido para restablecer la contraseña" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1445 +#, python-format +msgid "You are now logged in as: '%(nickname)s'" +msgstr "Ahora estás conectado como: '%(nickname)s'" + +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Perfil de %(name)s" -#: cps/web.py:1511 -#, fuzzy +#: cps/web.py:1526 msgid "Success! Profile Updated" msgstr "Perfil actualizado" -#: cps/web.py:1515 -#, fuzzy +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." -msgstr "Encontrada una cuenta existente para esa dirección de correo electrónico" +msgstr "Ya existe una cuenta creada para ese correo electrónico." #: cps/services/gmail.py:59 msgid "Found no valid gmail.json file with OAuth information" msgstr "No se ha encontrado ningún archivo gmail.json válido con información OAuth" -#: cps/tasks/convert.py:108 -#, fuzzy, python-format -msgid "%(book)s send to E-Reader" -msgstr "Enviar al Kindle" +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "Eliminar el contenido de la carpeta temporal" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:112 +#, python-format +msgid "%(book)s send to E-Reader" +msgstr "%(book)s enviado al eReader" + +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre ebook-convert %(tool)s no encontrado" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s format no encontrado en disco" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" -msgstr "El conversor de Ebook falló con un error desconocido" +msgstr "El conversor de ebook falló con un error desconocido" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-converter falló: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" -msgstr "Archivo convertido no encontrado, o más de un archivo en el directorio %(folder)s" +msgstr "Archivo convertido no encontrado, o hay más de un archivo en la carpeta %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Falló Ebook-converter: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre falló con el error: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Falló Ebook-converter: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" -msgstr "" +msgstr "Convertir fichero" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" -msgstr "" +msgstr "Reconectando la base de datos de Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" -msgstr "" +msgstr "Correo electrónico" #: cps/tasks/metadata_backup.py:34 -#, fuzzy msgid "Backing up Metadata" -msgstr "editar metadatos" - -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" +msgstr "Haciendo copia de seguridad de los metadatos" #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" -msgstr "" +msgstr "Se generaron %(count)s miniaturas de portada" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" -msgstr "" +msgstr "Miniaturas de portada" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" -msgstr "" +msgstr "Se generaron {0} miniaturas de series" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" -msgstr "" +msgstr "Borrando la caché de miniaturas de portada" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Subir archivo" @@ -1607,23 +1564,22 @@ msgstr "Nombre de usuario" msgid "Email" msgstr "Correo electrónico" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 -#, fuzzy +#: cps/templates/admin.html:15 msgid "Send to eReader Email" -msgstr "Enviar al correo de Kindle" +msgstr "Enviar al correo del eReader" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" -msgstr "Admin" +msgstr "Administrador" #: cps/templates/admin.html:18 cps/templates/login.html:13 #: cps/templates/login.html:14 cps/templates/user_edit.html:23 msgid "Password" msgstr "Contraseña" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Descargar" @@ -1666,7 +1622,7 @@ msgstr "Puerto SMTP" #: cps/templates/admin.html:75 cps/templates/email_edit.html:39 msgid "Encryption" -msgstr "Encriptado" +msgstr "Cifrado" #: cps/templates/admin.html:79 cps/templates/email_edit.html:47 msgid "SMTP Login" @@ -1675,16 +1631,15 @@ msgstr "Login SMTP" #: cps/templates/admin.html:83 cps/templates/admin.html:94 #: cps/templates/email_edit.html:55 msgid "From Email" -msgstr "Desde el correo" +msgstr "Desde el correo electrónico" #: cps/templates/admin.html:90 -#, fuzzy msgid "Email Service" -msgstr "Servicio de Correo" +msgstr "Servicio de correo electrónico" #: cps/templates/admin.html:91 msgid "Gmail via Oauth2" -msgstr "Gmail por Oauth2" +msgstr "Gmail a través de Oauth2" #: cps/templates/admin.html:106 msgid "Configuration" @@ -1735,7 +1690,6 @@ msgid "Reverse Proxy Header Name" msgstr "Nombre de cabecera de Proxy inverso" #: cps/templates/admin.html:159 -#, fuzzy msgid "Edit Calibre Database Configuration" msgstr "Editar la configuración de la base de datos de Calibre" @@ -1749,37 +1703,37 @@ msgstr "Editar la configuración de la interfaz de usuario" #: cps/templates/admin.html:167 msgid "Scheduled Tasks" -msgstr "" +msgstr "Tareas programadas" #: cps/templates/admin.html:170 cps/templates/schedule_edit.html:12 #: cps/templates/tasks.html:18 msgid "Start Time" -msgstr "" +msgstr "Hora de inicio" #: cps/templates/admin.html:174 cps/templates/schedule_edit.html:20 msgid "Maximum Duration" -msgstr "" +msgstr "Duración maxima" #: cps/templates/admin.html:178 cps/templates/schedule_edit.html:29 msgid "Generate Thumbnails" -msgstr "" +msgstr "Generar miniaturas" #: cps/templates/admin.html:182 msgid "Generate series cover thumbnails" -msgstr "" +msgstr "Generar miniaturas de portada de series" #: cps/templates/admin.html:186 cps/templates/admin.html:208 #: cps/templates/schedule_edit.html:37 msgid "Reconnect Calibre Database" -msgstr "" +msgstr "Reconectar la base de datos de Calibre" #: cps/templates/admin.html:190 cps/templates/schedule_edit.html:41 msgid "Generate Metadata Backup Files" -msgstr "" +msgstr "Generar archivos de copia de seguridad de metadatos" #: cps/templates/admin.html:197 msgid "Refresh Thumbnail Cache" -msgstr "" +msgstr "Actualizar la caché de miniaturas" #: cps/templates/admin.html:203 msgid "Administration" @@ -1803,7 +1757,7 @@ msgstr "Apagar" #: cps/templates/admin.html:221 msgid "Version Information" -msgstr "" +msgstr "Información de versión" #: cps/templates/admin.html:225 msgid "Version" @@ -1827,59 +1781,59 @@ msgstr "Realizar actualización" #: cps/templates/admin.html:253 msgid "Are you sure you want to restart?" -msgstr "¿Realmente quieres reiniciar?" +msgstr "¿Estás seguro de que deseas reiniciar?" #: cps/templates/admin.html:258 cps/templates/admin.html:272 #: cps/templates/admin.html:292 cps/templates/config_db.html:82 msgid "OK" -msgstr "Ok" +msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Cancelar" #: cps/templates/admin.html:271 msgid "Are you sure you want to shutdown?" -msgstr "¿Realmente quiere detener?" +msgstr "¿Estás seguro de que deseas apagar?" #: cps/templates/admin.html:283 msgid "Updating, please do not reload this page" -msgstr "Actualizando. Por favor, no recargue la página" +msgstr "Actualizando, por favor no recargues esta página" #: cps/templates/author.html:15 msgid "via" -msgstr "via" +msgstr "a través de" #: cps/templates/author.html:23 msgid "In Library" -msgstr "En la Librería" +msgstr "En la biblioteca" #: cps/templates/author.html:26 cps/templates/index.html:74 #: cps/templates/search.html:31 cps/templates/shelf.html:20 msgid "Sort according to book date, newest first" -msgstr "Ordenar en base a fecha del libro, más reciente primero" +msgstr "Ordenar en base a fecha de subida del libro, más reciente primero" #: cps/templates/author.html:27 cps/templates/index.html:75 #: cps/templates/search.html:32 cps/templates/shelf.html:21 msgid "Sort according to book date, oldest first" -msgstr "Ordenar en base a fecha del libro, menos reciente primero" +msgstr "Ordenar en base a fecha de subida del libro, más antiguo primero" #: cps/templates/author.html:28 cps/templates/index.html:76 #: cps/templates/search.html:33 cps/templates/shelf.html:22 msgid "Sort title in alphabetical order" -msgstr "Ordenar en base al título en orden alfabético" +msgstr "Ordenar en base al título en orden alfabético ascendente" #: cps/templates/author.html:29 cps/templates/index.html:77 #: cps/templates/search.html:34 cps/templates/shelf.html:23 msgid "Sort title in reverse alphabetical order" -msgstr "Ordenar en base al título en orden alfabético invertido" +msgstr "Ordenar en base al título en orden alfabético descendente" #: cps/templates/author.html:30 cps/templates/index.html:80 #: cps/templates/search.html:37 cps/templates/shelf.html:26 @@ -1889,18 +1843,78 @@ msgstr "Ordenar en base a fecha de publicación, más reciente primero" #: cps/templates/author.html:31 cps/templates/index.html:81 #: cps/templates/search.html:38 cps/templates/shelf.html:27 msgid "Sort according to publishing date, oldest first" -msgstr "Ordenar en base a fecha de publicación, menos reciente primero" +msgstr "Ordenar en base a fecha de publicación, más antiguo primero" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "reducir" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Más de" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Libro %(index)s de %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Idioma" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Editor" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publicado" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Descripción:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Anterior" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Siguiente" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "No se han encontrado resultados" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Inicio" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Buscar en la biblioteca" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Cerrar sesión" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Borrar libro" @@ -1929,99 +1943,107 @@ msgstr "Convertir a:" msgid "Convert book" msgstr "Convertir libro" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Subiendo..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Cerrar" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Error" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Carga completada, procesando, por favor espera..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Subir formato" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Título del libro" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Etiquetas" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID de serie" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Fecha de publicación" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Valoración" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Descripción" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identificadores" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Tipo de identificador" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Valor de identificador" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Borrar" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Añadir identificador" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Etiquetas" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID de serie" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Clasificación" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" -msgstr "Obtener portada de URL (JPEG, la portada ser'a descargada y almacenada en la base de datos)" +msgstr "Obtener portada desde URL (JPEG - La imagen será descargada y almacenada en la base de datos)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Subir portada desde un disco local" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Fecha de publicación" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Editor" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Idioma" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Sí" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "No" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Subir formato" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" -msgstr "Ver libro tras la edición" +msgstr "Ver libro al guardar" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Obtener metadatos" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2029,38 +2051,31 @@ msgstr "Obtener metadatos" msgid "Save" msgstr "Guardar" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Palabra clave" -#: cps/templates/book_edit.html:234 -#, fuzzy -msgid "Search keyword" -msgstr " Buscar por palabras clave " - #: cps/templates/book_edit.html:240 +msgid "Search keyword" +msgstr "Buscar por palabras clave" + +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Haz clic en la portada para cargar los metadatos en el formulario" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Cargando..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Cerrar" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Origen" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "¡Error en la búsqueda!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "¡No se encontraron resultados! Por favor intenta con otra palabra clave." @@ -2144,42 +2159,40 @@ msgid "Enter Publishers" msgstr "Introduce los Editores" #: cps/templates/book_table.html:73 -#, fuzzy msgid "Enter comments" -msgstr "Introducir nombre de dominio" +msgstr "Introduce comentarios" #: cps/templates/book_table.html:73 msgid "Comments" -msgstr "" +msgstr "Comentarios" #: cps/templates/book_table.html:75 msgid "Archive Status" -msgstr "" +msgstr "Archivado" #: cps/templates/book_table.html:77 cps/templates/search_form.html:42 msgid "Read Status" -msgstr "Leer estado" +msgstr "Leído" #: cps/templates/book_table.html:80 cps/templates/book_table.html:82 #: cps/templates/book_table.html:84 cps/templates/book_table.html:86 #: cps/templates/book_table.html:90 cps/templates/book_table.html:92 #: cps/templates/book_table.html:96 -#, fuzzy msgid "Enter " -msgstr "Identificadores" +msgstr "Introducir " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "¿Estás realmente seguro?" #: cps/templates/book_table.html:117 msgid "Books with Title will be merged from:" -msgstr "Libros con título serán fusionados de:" +msgstr "Los libros con el título serán fusionados de:" #: cps/templates/book_table.html:121 msgid "Into Book with Title:" -msgstr "En el libro con el título:" +msgstr "Al libro con el título:" #: cps/templates/book_table.html:126 msgid "Merge" @@ -2191,7 +2204,7 @@ msgstr "Ubicación de la base de datos Calibre" #: cps/templates/config_db.html:21 msgid "Separate Book Files from Library" -msgstr "" +msgstr "Separar archivos de libros de la biblioteca" #: cps/templates/config_db.html:34 msgid "Use Google Drive?" @@ -2199,7 +2212,7 @@ msgstr "¿Usar Google Drive?" #: cps/templates/config_db.html:39 msgid "Authenticate Google Drive" -msgstr "Autentificar Google Drive" +msgstr "Autenticar Google Drive" #: cps/templates/config_db.html:44 msgid "Google Drive Calibre folder" @@ -2214,9 +2227,8 @@ msgid "Revoke" msgstr "Revocar" #: cps/templates/config_db.html:80 -#, fuzzy msgid "New db location is invalid, please enter valid path" -msgstr "La ruta de la base de datos no es válida. Por favor, Introduce la ruta correcta" +msgstr "La nueva ubicación de la base de datos no es válida, por favor introduce una ruta válida" #: cps/templates/config_edit.html:18 msgid "Server Configuration" @@ -2248,7 +2260,7 @@ msgstr "Nocturno" #: cps/templates/config_edit.html:50 msgid "Trusted Hosts (Comma Separated)" -msgstr "" +msgstr "Hosts de confianza (separados por comas)" #: cps/templates/config_edit.html:61 msgid "Logfile Configuration" @@ -2272,11 +2284,11 @@ msgstr "Configuración de características" #: cps/templates/config_edit.html:104 msgid "Convert non-English characters in title and author while saving to disk" -msgstr "" +msgstr "Convertir caracteres no ingleses en el título y autor al guardar en disco" #: cps/templates/config_edit.html:108 msgid "Embed Metadata to Ebook File on Download/Conversion/e-mail (needs Calibre/Kepubify binaries)" -msgstr "" +msgstr "Incrustar metadatos en el archivo del libro electrónico al descargar/converter/enviar por correo electrónico (necesita los binarios de Calibre/Kepubify)" #: cps/templates/config_edit.html:112 msgid "Enable Uploads" @@ -2284,7 +2296,7 @@ msgstr "Permitir subidas" #: cps/templates/config_edit.html:112 msgid "(Please ensure that users also have upload permissions)" -msgstr "" +msgstr "(Por favor, asegúrate de que los usuarios también tengan permisos de carga)" #: cps/templates/config_edit.html:116 msgid "Allowed Upload Fileformats" @@ -2300,11 +2312,11 @@ msgstr "Permitir registro público" #: cps/templates/config_edit.html:131 msgid "Use Email as Username" -msgstr "Utilizar eMail como nombre de usuario" +msgstr "Usar el correo electrónico como nombre de usuario" #: cps/templates/config_edit.html:136 msgid "Enable Magic Link Remote Login" -msgstr "Permitir inicio de sesión remoto (\"magic link\")" +msgstr "Permitir inicio de sesión remoto (\"Magic Link\")" #: cps/templates/config_edit.html:141 msgid "Enable Kobo sync" @@ -2316,7 +2328,7 @@ msgstr "Peticiones proxy a la tienda Kobo desconocidas" #: cps/templates/config_edit.html:149 msgid "Server External Port (for port forwarded API calls)" -msgstr "Puerto externo del servidor (para peticiones API)" +msgstr "Puerto externo del servidor (para llamadas API con reenvío de puerto)" #: cps/templates/config_edit.html:157 msgid "Use Goodreads" @@ -2324,11 +2336,11 @@ msgstr "Usar Goodreads" #: cps/templates/config_edit.html:161 msgid "Goodreads API Key" -msgstr "Goodreads API Key" +msgstr "Clave de API de Goodreads" #: cps/templates/config_edit.html:168 msgid "Allow Reverse Proxy Authentication" -msgstr "Permitir Autenticación Proxy Inversas" +msgstr "Permitir autenticación de proxy inverso" #: cps/templates/config_edit.html:179 msgid "Login type" @@ -2356,7 +2368,7 @@ msgstr "Puerto del servidor LDAP" #: cps/templates/config_edit.html:201 msgid "LDAP Encryption" -msgstr "Encriptación LDAP" +msgstr "Cifrado LDAP" #: cps/templates/config_edit.html:204 msgid "TLS" @@ -2368,15 +2380,15 @@ msgstr "SSL" #: cps/templates/config_edit.html:209 msgid "LDAP CACertificate Path (Only needed for Client Certificate Authentication)" -msgstr "Ruta LDAP CACertificate (Solo necesaria para certificado de autenticación de cliente)" +msgstr "Ruta del certificado CA de LDAP (solo necesario para la autenticación con certificado de cliente)" #: cps/templates/config_edit.html:216 msgid "LDAP Certificate Path (Only needed for Client Certificate Authentication)" -msgstr "Ruta LDAP Certificate (Solo necesaria para certificado de autenticación de cliente)" +msgstr "Ruta del certificado de LDAP (solo necesario para la autenticación con certificado de cliente)" #: cps/templates/config_edit.html:223 msgid "LDAP Keyfile Path (Only needed for Client Certificate Authentication)" -msgstr "Ruta LDAP Keyfile (Solo necesaria para certificado de autenticación de cliente)" +msgstr "Ruta del archivo de clave LDAP (solo necesario para la autenticación con certificado de cliente)" #: cps/templates/config_edit.html:232 msgid "LDAP Authentication" @@ -2404,7 +2416,7 @@ msgstr "Contraseña de administrador LDAP" #: cps/templates/config_edit.html:252 msgid "LDAP Distinguished Name (DN)" -msgstr "Nombre distinguido LDAP (DN)" +msgstr "Nombre distintivo LDAP (DN)" #: cps/templates/config_edit.html:256 msgid "LDAP User Object Filter" @@ -2416,7 +2428,7 @@ msgstr "¿El servidor LDAP es OpenLDAP?" #: cps/templates/config_edit.html:263 msgid "Following Settings are Needed For User Import" -msgstr "La siguiente configuración son necesarias para la importación de usuarios" +msgstr "Se necesitan las siguientes configuraciones para la importación de usuarios" #: cps/templates/config_edit.html:265 msgid "LDAP Group Object Filter" @@ -2432,7 +2444,7 @@ msgstr "Campo de miembros de grupo LDAP" #: cps/templates/config_edit.html:277 msgid "LDAP Member User Filter Detection" -msgstr "Filtro de detección LDAP Member User" +msgstr "Detección de filtro de usuarios miembros de LDAP" #: cps/templates/config_edit.html:279 msgid "Autodetect" @@ -2444,31 +2456,30 @@ msgstr "Filtro personalizado" #: cps/templates/config_edit.html:285 msgid "LDAP Member User Filter" -msgstr "Filtro LDAP Member User" +msgstr "Filtro de usuarios miembros de LDAP" #: cps/templates/config_edit.html:296 #, python-format msgid "Obtain %(provider)s OAuth Credential" -msgstr "Obtener la Credencial OAuth de %(provider)s" +msgstr "Obtener las credenciales OAuth de %(provider)s" #: cps/templates/config_edit.html:299 #, python-format msgid "%(provider)s OAuth Client Id" -msgstr "Id de cliente de OAuth de %(provider)s" +msgstr "ID de cliente OAuth de %(provider)s" #: cps/templates/config_edit.html:303 #, python-format msgid "%(provider)s OAuth Client Secret" -msgstr "Secreto OAuth de Cliente de %(provider)s" +msgstr "Secreto de cliente OAuth (OAuth Client Secret) de %(provider)s" #: cps/templates/config_edit.html:319 msgid "External binaries" msgstr "Binarios externos" #: cps/templates/config_edit.html:325 -#, fuzzy msgid "Path to Calibre Binaries" -msgstr "Ruta para Calibre E-Book Converter" +msgstr "Ruta a los binarios de Calibre" #: cps/templates/config_edit.html:333 msgid "Calibre E-Book Converter Settings" @@ -2479,71 +2490,68 @@ msgid "Path to Kepubify E-Book Converter" msgstr "Ruta para Kepubify E-Book Converter" #: cps/templates/config_edit.html:344 -#, fuzzy msgid "Location of Unrar binary" msgstr "Ubicación del binario de UnRar" #: cps/templates/config_edit.html:360 -#, fuzzy msgid "Security Settings" -msgstr "Ajustes OAuth" +msgstr "Configuraciones de seguridad" #: cps/templates/config_edit.html:368 msgid "Limit failed login attempts" -msgstr "" +msgstr "Limitar los intentos de inicio de sesión fallidos" #: cps/templates/config_edit.html:372 msgid "Configure Backend for Limiter" -msgstr "" +msgstr "Configurar backend para Limiter" #: cps/templates/config_edit.html:376 msgid "Options for Limiter Backend" -msgstr "" +msgstr "Opciones del backend de Limiter" #: cps/templates/config_edit.html:382 msgid "Check if file extensions matches file content on upload" -msgstr "" +msgstr "Verificar si las extensiones de archivo coinciden con el contenido del archivo al cargar" #: cps/templates/config_edit.html:385 msgid "Session protection" -msgstr "" +msgstr "Protección de sesión" #: cps/templates/config_edit.html:387 msgid "Basic" -msgstr "" +msgstr "Débil" #: cps/templates/config_edit.html:388 msgid "Strong" -msgstr "" +msgstr "Fuerte" #: cps/templates/config_edit.html:393 -#, fuzzy msgid "User Password policy" -msgstr "Resetear contraseña de usuario" +msgstr "Política de contraseñas de usuario" #: cps/templates/config_edit.html:397 msgid "Minimum password length" -msgstr "" +msgstr "Longitud mínima de la contraseña" #: cps/templates/config_edit.html:402 msgid "Enforce number" -msgstr "" +msgstr "Forzar números" #: cps/templates/config_edit.html:406 msgid "Enforce lowercase characters" -msgstr "" +msgstr "Forzar caracteres en minúscula" #: cps/templates/config_edit.html:410 msgid "Enforce uppercase characters" -msgstr "" +msgstr "Forzar caracteres en mayuscula" #: cps/templates/config_edit.html:414 msgid "Enforce characters (needed For Chinese/Japanese/Korean Characters)" -msgstr "" +msgstr "Forzar caracteres (necesario para caracteres chinos/japoneses/coreanos)" #: cps/templates/config_edit.html:418 msgid "Enforce special characters" -msgstr "" +msgstr "Forzar caracteres especiales" #: cps/templates/config_view_edit.html:17 msgid "View Configuration" @@ -2567,7 +2575,7 @@ msgstr "Tema estándar" #: cps/templates/config_view_edit.html:43 msgid "caliBlur! Dark Theme" -msgstr "caliBlur! Tema Oscuro" +msgstr "caliBlur! Tema oscuro" #: cps/templates/config_view_edit.html:47 msgid "Regular Expression for Ignoring Columns" @@ -2622,14 +2630,12 @@ msgid "Allow Editing Public Shelves" msgstr "Permitir editar estantes públicos" #: cps/templates/config_view_edit.html:123 -#, fuzzy msgid "Default Language" -msgstr "Excluir idiomas" +msgstr "Idioma predeterminado" #: cps/templates/config_view_edit.html:131 -#, fuzzy msgid "Default Visible Language of Books" -msgstr "Mostrar libros con idioma" +msgstr "Visibilidad de idioma de los libros predeterminada" #: cps/templates/config_view_edit.html:147 msgid "Default Visibilities for New Users" @@ -2642,80 +2648,65 @@ msgstr "Mostrar libros aleatorios en la vista detallada" #: cps/templates/config_view_edit.html:166 cps/templates/user_edit.html:87 msgid "Add Allowed/Denied Tags" -msgstr "Añadir etiquetas Permitidas/Denegados" +msgstr "Añadir etiquetas permitidas/prohibidas" #: cps/templates/config_view_edit.html:167 msgid "Add Allowed/Denied custom column values" -msgstr "Añadir valores personalizados Permitidos/Denegados" +msgstr "Añadir valores personalizados permitidos/prohibidos" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Leer en el navegador" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Escuchar en el navegador" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publicado" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Marcar como no leido" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Marcar como leido" -#: cps/templates/detail.html:254 -#, fuzzy +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" -msgstr "Marcar como no leido" +msgstr "Marcar como leído o no leido" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Leído" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Restarurar desde el archivo" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" -msgstr "Añadir a archivación" +msgstr "Añadir a archivo" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" -msgstr "" +msgstr "Marcar el libro como archivado o no, para ocultarlo en Calibre-Web y eliminarlo del lector Kobo" -#: cps/templates/detail.html:267 -#, fuzzy +#: cps/templates/detail.html:275 msgid "Archive" -msgstr "Archivado" +msgstr "Archivar" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Descripción:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" -msgstr "Agregar al estante" +msgstr "Agregar a la estantería" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" -msgstr "(Público)" +msgstr "(Pública)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Editar metadatos" @@ -2724,18 +2715,16 @@ msgid "Email Account Type" msgstr "Elige tipo de servidor" #: cps/templates/email_edit.html:15 -#, fuzzy msgid "Standard Email Account" msgstr "Usar cuenta de correo estándar" #: cps/templates/email_edit.html:16 -#, fuzzy msgid "Gmail Account" -msgstr "Elige tipo de servidor" +msgstr "Cuenta de Gmail" #: cps/templates/email_edit.html:22 msgid "Setup Gmail Account" -msgstr "" +msgstr "Configurar cuenta de Gmail" #: cps/templates/email_edit.html:24 msgid "Revoke Gmail Access" @@ -2755,10 +2744,9 @@ msgstr "Contraseña SMTP" #: cps/templates/email_edit.html:58 msgid "Attachment Size Limit" -msgstr "Tamaño límite del archivo adjunto" +msgstr "Límite de tamaño de archivo adjunto" #: cps/templates/email_edit.html:66 -#, fuzzy msgid "Save and Send Test Email" msgstr "Guardar y enviar un correo electrónico de prueba" @@ -2769,7 +2757,7 @@ msgstr "Regresar" #: cps/templates/email_edit.html:74 msgid "Allowed Domains (Whitelist)" -msgstr "Dominios permitidos para registrarse" +msgstr "Dominios permitidos (Lista blanca)" #: cps/templates/email_edit.html:78 cps/templates/email_edit.html:105 msgid "Add Domain" @@ -2786,59 +2774,58 @@ msgstr "Introducir nombre de dominio" #: cps/templates/email_edit.html:92 msgid "Denied Domains (Blacklist)" -msgstr "Dominios prohibidos (Blaclist)" - -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Siguiente" +msgstr "Dominios prohibidos (Lista negra)" #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Abre el archivo .kobo/Kobo/Kobo eReader.conf en un editor de texto y añade (o edita):" #: cps/templates/generate_kobo_auth_url.html:11 -#, fuzzy msgid "Kobo Token:" -msgstr "Token de sincronización de Kobo" +msgstr "Token de sincronización de Kobo:" #: cps/templates/grid.html:21 msgid "List" -msgstr "" +msgstr "Lista" #: cps/templates/http_error.html:34 -#, fuzzy msgid "Calibre-Web Instance is unconfigured, please contact your administrator" -msgstr "El servidor de E-Mail no está configurado, por favor contacta a tu administrador" +msgstr "La instancia de Calibre-Web no está configurada, por favor contacta con tu administrador" #: cps/templates/http_error.html:44 msgid "Create Issue" msgstr "Crear una incidencia" -#: cps/templates/http_error.html:51 -msgid "Return to Home" -msgstr "Volver al inicio" +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Configuración de la base de datos" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:54 +msgid "Return to Home" +msgstr "Volver a inicio" + +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Cerrar sesión" #: cps/templates/index.html:71 msgid "Sort ascending according to download count" -msgstr "" +msgstr "Ordenar ascendientemente según el número de descargas" #: cps/templates/index.html:72 msgid "Sort descending according to download count" -msgstr "" +msgstr "Ordenar descendientemente según el número de descargas" #: cps/templates/index.html:78 cps/templates/search.html:35 #: cps/templates/shelf.html:24 msgid "Sort authors in alphabetical order" -msgstr "Ordenar autores en orden alfabético" +msgstr "Ordenar en base al autor en orden alfabético ascendente" #: cps/templates/index.html:79 cps/templates/search.html:36 #: cps/templates/shelf.html:25 msgid "Sort authors in reverse alphabetical order" -msgstr "Ordenar autores en orden alfabético inverso" +msgstr "Ordenar en base al autor en orden alfabético descendente" #: cps/templates/index.html:83 msgid "Sort ascending according to series index" @@ -2854,7 +2841,7 @@ msgstr "Iniciar" #: cps/templates/index.xml:19 msgid "Alphabetical Books" -msgstr "Libros Alfabéticos" +msgstr "Libros alfabéticos" #: cps/templates/index.xml:23 msgid "Books sorted alphabetically" @@ -2862,11 +2849,11 @@ msgstr "Libros ordenados alfabéticamente" #: cps/templates/index.xml:31 msgid "Popular publications from this catalog based on Downloads." -msgstr "Publicaciones populares de este catálogo basadas en las Descargas." +msgstr "Publicaciones populares de este catálogo basadas en las descargas." #: cps/templates/index.xml:40 msgid "Popular publications from this catalog based on Rating." -msgstr "Publicaciones populares del catálogo basados en la clasificación." +msgstr "Publicaciones populares de este catálogo basadas en la valoración." #: cps/templates/index.xml:45 msgid "Recently added Books" @@ -2874,7 +2861,7 @@ msgstr "Libros añadidos recientemente" #: cps/templates/index.xml:49 msgid "The latest Books" -msgstr "Últimos ibros" +msgstr "Últimos libros" #: cps/templates/index.xml:54 msgid "Random Books" @@ -2902,13 +2889,13 @@ msgstr "Libros ordenados por idioma" #: cps/templates/index.xml:128 msgid "Books ordered by Rating" -msgstr "Libros ordenados por puntuación" +msgstr "Libros ordenados por valoración" #: cps/templates/index.xml:137 msgid "Books ordered by file formats" msgstr "Libros ordenados por formato de archivo" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Estanterías" @@ -2917,66 +2904,43 @@ msgstr "Estanterías" msgid "Books organized in shelves" msgstr "Libros organizados en estanterías" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Inicio" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Alternar navegación" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Buscar en la librería" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Simple" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Cuenta" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Cerrar sesión" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Cargando..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Error" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Carga hecha, procesando, por favor espere ..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Ajustes" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" -msgstr "Por favor, no actualice la página" +msgstr "Por favor, no actualices la página" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Navegar" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Acerca de" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Previo" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Detalles del libro" #: cps/templates/list.html:22 msgid "Grid" -msgstr "" +msgstr "Cuadricula" #: cps/templates/listenmp3.html:167 msgid "Archived" @@ -2988,11 +2952,11 @@ msgstr "Recordarme" #: cps/templates/login.html:23 msgid "Forgot Password?" -msgstr "¿Olvidó la contraseña?" +msgstr "¿Olvidaste tu contraseña?" #: cps/templates/login.html:34 msgid "Log in with Magic Link" -msgstr "Iniciar sesión con un enlace mágico" +msgstr "Iniciar sesión con un enlace mágico (Magic Link)" #: cps/templates/logviewer.html:6 msgid "Show Calibre-Web Log: " @@ -3004,11 +2968,11 @@ msgstr "Registro de Calibre-Web: " #: cps/templates/logviewer.html:8 msgid "Stream output, can't be displayed" -msgstr "La salida del flujo no pudo ser mostrada" +msgstr "Salida de transmisión, no se puede mostrar" #: cps/templates/logviewer.html:12 msgid "Show Access Log: " -msgstr "Mostrar registro de acceso:" +msgstr "Mostrar registro de acceso: " #: cps/templates/logviewer.html:18 msgid "Download Calibre-Web Log" @@ -3020,23 +2984,23 @@ msgstr "Descargar el registro de acceso" #: cps/templates/modal_dialogs.html:6 msgid "Select Allowed/Denied Tags" -msgstr "Seleccionar etiquetas Permitidas/Denegadas" +msgstr "Seleccionar etiquetas permitidas/prohibidas" #: cps/templates/modal_dialogs.html:7 msgid "Select Allowed/Denied Custom Column Values" -msgstr "Seleccionar valores propios de columnas Permitidas/Denegadas" +msgstr "Seleccionar columnas personalizadas permitidas/prohibidas" #: cps/templates/modal_dialogs.html:8 msgid "Select Allowed/Denied Tags of User" -msgstr "Seleccionar etiquetas de usuario Permitido/Denegado" +msgstr "Seleccionar etiquetas de usuario permitido/prohibido" #: cps/templates/modal_dialogs.html:9 msgid "Select Allowed/Denied Custom Column Values of User" -msgstr "Seleccionar columnas personalizadas de usuario Permitido/Denegado" +msgstr "Seleccionar columnas personalizadas de usuario permitidas/prohibidas" #: cps/templates/modal_dialogs.html:15 msgid "Enter Tag" -msgstr "Introduce la etiqueta" +msgstr "Introduce etiqueta" #: cps/templates/modal_dialogs.html:24 msgid "Add View Restriction" @@ -3044,7 +3008,7 @@ msgstr "Añadir restricción de vista" #: cps/templates/modal_dialogs.html:50 msgid "This book format will be permanently erased from database" -msgstr "Este libro será borrado permanentemente de la base de datos" +msgstr "Este libro será eliminado permanentemente de la base de datos" #: cps/templates/modal_dialogs.html:51 msgid "This book will be permanently erased from database" @@ -3060,7 +3024,7 @@ msgstr "Nota de Kobo importante: los libros eliminados permanecerán en los disp #: cps/templates/modal_dialogs.html:57 msgid "Books must first be archived and the device synced before a book can safely be deleted." -msgstr "Antes de que un libro pueda ser eliminado con seguridad debe ser archivado y sincronixado con el dispositivo." +msgstr "Antes de que un libro pueda ser eliminado con seguridad debe ser archivado y sincronizado con el dispositivo." #: cps/templates/modal_dialogs.html:76 msgid "Choose File Location" @@ -3086,8 +3050,7 @@ msgstr "Directorio padre" msgid "Select" msgstr "Seleccionar" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 -#, fuzzy +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3095,40 +3058,77 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Catálogo de ebooks de Calibre-Web" -#: cps/templates/read.html:6 -#, fuzzy +#: cps/templates/read.html:7 msgid "epub Reader" -msgstr "Lector PDF" +msgstr "Lector epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +msgid "Choose a theme below:" +msgstr "Elije un tema:" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Claro" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Oscuro" -#: cps/templates/read.html:83 -msgid "Sepia" -msgstr "" - -#: cps/templates/read.html:84 -#, fuzzy -msgid "Black" -msgstr "Regresar" - #: cps/templates/read.html:88 -msgid "Reflow text when sidebars are open." -msgstr "Redimensionar el texto cuando las barras laterales están abiertas." +msgid "Sepia" +msgstr "Sepia" -#: cps/templates/read.html:93 +#: cps/templates/read.html:90 +msgid "Black" +msgstr "Negro" + +#: cps/templates/read.html:95 +msgid "Reflow text when sidebars are open." +msgstr "Reformatear el texto cuando las barras laterales están abiertas." + +#: cps/templates/read.html:100 msgid "Font Sizes" -msgstr "" +msgstr "Tamaños de la fuente" + +#: cps/templates/read.html:105 +msgid "Font" +msgstr "Fuente" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "Predeterminada" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "Yahei" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "SimSun" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "KaiTi" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "Extensión" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "Dos columnas" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "Una columna" #: cps/templates/readcbr.html:8 -#, fuzzy msgid "Comic Reader" -msgstr "Lector PDF" +msgstr "Lector de comics" #: cps/templates/readcbr.html:75 msgid "Keyboard Shortcuts" @@ -3136,7 +3136,7 @@ msgstr "Atajos de teclado" #: cps/templates/readcbr.html:78 msgid "Previous Page" -msgstr "Página previa" +msgstr "Página anterior" #: cps/templates/readcbr.html:79 cps/templates/readcbr.html:159 msgid "Next Page" @@ -3144,11 +3144,11 @@ msgstr "Página siguiente" #: cps/templates/readcbr.html:80 msgid "Single Page Display" -msgstr "" +msgstr "Visualización de página única" #: cps/templates/readcbr.html:81 msgid "Long Strip Display" -msgstr "" +msgstr "Visualización de tira larga" #: cps/templates/readcbr.html:82 msgid "Scale to Best" @@ -3156,7 +3156,7 @@ msgstr "Escalar a mejor" #: cps/templates/readcbr.html:83 msgid "Scale to Width" -msgstr "Escalar a la ancho" +msgstr "Escalar a lo ancho" #: cps/templates/readcbr.html:84 msgid "Scale to Height" @@ -3180,16 +3180,15 @@ msgstr "Voltear imagen" #: cps/templates/readcbr.html:110 msgid "Display" -msgstr "" +msgstr "Mostrar" #: cps/templates/readcbr.html:113 -#, fuzzy msgid "Single Page" -msgstr "Página de administración" +msgstr "Página única" #: cps/templates/readcbr.html:114 msgid "Long Strip" -msgstr "" +msgstr "Tira larga" #: cps/templates/readcbr.html:119 msgid "Scale" @@ -3241,11 +3240,11 @@ msgstr "De derecha a izquierda" #: cps/templates/readcbr.html:162 msgid "Reset to Top" -msgstr "" +msgstr "Restablecer a la parte superior" #: cps/templates/readcbr.html:163 msgid "Remember Position" -msgstr "" +msgstr "Recordar posición" #: cps/templates/readcbr.html:168 msgid "Scrollbar" @@ -3260,55 +3259,48 @@ msgid "Hide" msgstr "Ocultar" #: cps/templates/readdjvu.html:5 -#, fuzzy msgid "DJVU Reader" -msgstr "Lector PDF" +msgstr "Lector DJVU" #: cps/templates/readpdf.html:31 -#, fuzzy msgid "PDF Reader" msgstr "Lector PDF" #: cps/templates/readtxt.html:6 -#, fuzzy msgid "txt Reader" -msgstr "Lector PDF" +msgstr "Lector de ficheros de texto (.txt)" #: cps/templates/register.html:4 msgid "Register New Account" -msgstr "Registre una cuenta nueva" +msgstr "Registrar nueva cuenta" #: cps/templates/register.html:10 msgid "Choose a username" -msgstr "Escoger un nombre de usuario" +msgstr "Elige un nombre de usuario" #: cps/templates/register.html:15 msgid "Your Email" -msgstr "Tu dirección de correo" +msgstr "Dirección de correo electrónico" #: cps/templates/remote_login.html:5 msgid "Magic Link - Authorise New Device" -msgstr "Enlace Mágico - Autorizar un nuevo dispositivo" +msgstr "Enlace mágico - Autorizar un nuevo dispositivo" #: cps/templates/remote_login.html:7 msgid "On another device, login and visit:" -msgstr "Utiliza otro dispositivo, inicia sesión y visita:" +msgstr "En otro dispositivo, inicia sesión y visita:" #: cps/templates/remote_login.html:11 msgid "Once verified, you will automatically be logged in on this device." -msgstr "Una vez que lo realice, iniciará sesión automáticamente en ese dispositivo." +msgstr "Una vez verificado, iniciarás sesión automáticamente en este dispositivo." #: cps/templates/remote_login.html:14 msgid "This verification link will expire in 10 minutes." -msgstr "El enlace expirará después de 10 minutos." +msgstr "Este enlace de verificación expirará en 10 minutos." #: cps/templates/schedule_edit.html:33 msgid "Generate Series Cover Thumbnails" -msgstr "" - -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "No se han encontrado resultados" +msgstr "Generar miniaturas de portada de series" #: cps/templates/search.html:7 msgid "Search Term:" @@ -3326,13 +3318,13 @@ msgstr "Fecha de publicación desde" msgid "Published Date To" msgstr "Fecha de publicación hasta" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" -msgstr "" +msgstr "Cualquiera" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" -msgstr "" +msgstr "Vacío" #: cps/templates/search_form.html:60 msgid "Exclude Tags" @@ -3343,9 +3335,8 @@ msgid "Exclude Series" msgstr "Excluir series" #: cps/templates/search_form.html:96 -#, fuzzy msgid "Exclude Shelves" -msgstr "Excluir estantes" +msgstr "Excluir estanterias" #: cps/templates/search_form.html:116 msgid "Exclude Languages" @@ -3357,7 +3348,7 @@ msgstr "Extensiones" #: cps/templates/search_form.html:135 msgid "Exclude Extensions" -msgstr "Extensiones excluidas" +msgstr "Excluir extensiones" #: cps/templates/search_form.html:145 msgid "Rating Above" @@ -3367,21 +3358,23 @@ msgstr "Clasificación mayor que" msgid "Rating Below" msgstr "Clasificación menor que" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "De:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Para:" #: cps/templates/shelf.html:13 msgid "Delete this Shelf" -msgstr "Borrar este estante" +msgstr "Borrar esta estanteria" #: cps/templates/shelf.html:14 msgid "Edit Shelf Properties" -msgstr "Editar propiedades del estante" +msgstr "Editar propiedades de la estantería" #: cps/templates/shelf.html:17 msgid "Arrange books manually" @@ -3395,13 +3388,21 @@ msgstr "Deshabilitar cambio de orden" msgid "Enable Change order" msgstr "Habilitar cambio de orden" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "Ordenar según la fecha de agregado a la estantería, el más reciente primero" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "Ordenar según la fecha de agregado a la estantería, el más antiguo primero" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Compartir con todos" #: cps/templates/shelf_edit.html:21 msgid "Sync this shelf with Kobo device" -msgstr "Sincronizar este estante con un dispositivo Kobo" +msgstr "Sincronizar esta estanteria con un dispositivo Kobo" #: cps/templates/shelf_order.html:5 msgid "Drag to Rearrange Order" @@ -3437,7 +3438,7 @@ msgstr "Estadísticas del sistema" #: cps/templates/stats.html:33 msgid "Program" -msgstr "" +msgstr "Programa" #: cps/templates/stats.html:34 msgid "Installed Version" @@ -3463,29 +3464,37 @@ msgstr "Progreso" msgid "Run Time" msgstr "Tiempo de ejecución" -#: cps/templates/tasks.html:20 -msgid "Actions" -msgstr "" +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "Mensaje" -#: cps/templates/tasks.html:40 -msgid "This task will be cancelled. Any progress made by this task will be saved." -msgstr "" +#: cps/templates/tasks.html:21 +msgid "Actions" +msgstr "Acciones" #: cps/templates/tasks.html:41 +msgid "This task will be cancelled. Any progress made by this task will be saved." +msgstr "Esta tarea será cancelada. Cualquier progreso realizado por esta tarea será guardado." + +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." -msgstr "" +msgstr "Si esta es una tarea programada, se volverá a ejecutar durante el próximo horario programado." #: cps/templates/user_edit.html:20 msgid "Reset user Password" -msgstr "Resetear contraseña de usuario" +msgstr "Restablecer contraseña de usuario" + +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "Enviar a la dirección de correo electrónico del eReader. Usa comas para separar correos electrónicos para múltiples eReaders" #: cps/templates/user_edit.html:43 msgid "Language of Books" -msgstr "Mostrar libros con idioma" +msgstr "Idioma de los libros" #: cps/templates/user_edit.html:54 msgid "OAuth Settings" -msgstr "Ajustes OAuth" +msgstr "Ajustes de OAuth" #: cps/templates/user_edit.html:56 msgid "Link" @@ -3505,11 +3514,11 @@ msgstr "Crear/Ver" #: cps/templates/user_edit.html:70 msgid "Force full kobo sync" -msgstr "" +msgstr "Forzar sincronización completa de Kobo" #: cps/templates/user_edit.html:88 msgid "Add allowed/Denied Custom Column Values" -msgstr "Añadir columnas de valores propios de Permitidos/Denegados" +msgstr "Añadir columnas personalizadas de usuario permitidas/prohibidas" #: cps/templates/user_edit.html:137 msgid "Sync only books in selected shelves with Kobo" @@ -3528,32 +3537,26 @@ msgid "Select..." msgstr "Selección..." #: cps/templates/user_table.html:131 -#, fuzzy msgid "Edit User" msgstr "Editar usuario" #: cps/templates/user_table.html:134 -#, fuzzy msgid "Enter Username" -msgstr "Introduce el nombre de usuario" +msgstr "Introduce nombre de usuario" #: cps/templates/user_table.html:135 -#, fuzzy msgid "Enter Email" -msgstr "Comprobar correo electrónico" +msgstr "Introduce correo electrónico" #: cps/templates/user_table.html:136 -#, fuzzy msgid "Enter eReader Email" -msgstr "Enviar al correo de Kindle" +msgstr "Introduce correo electrónico del eReader" #: cps/templates/user_table.html:136 -#, fuzzy msgid "eReader Email" -msgstr "Comprobar correo electrónico" +msgstr "Correo electrónico del eReader" #: cps/templates/user_table.html:137 -#, fuzzy msgid "Locale" msgstr "Idioma" @@ -3562,45 +3565,38 @@ msgid "Visible Book Languages" msgstr "Idiomas de libros visibles" #: cps/templates/user_table.html:139 -#, fuzzy msgid "Edit Allowed Tags" -msgstr "Seleccionar etiquetas Permitidas" +msgstr "Editar etiquetas permitidas" #: cps/templates/user_table.html:139 msgid "Allowed Tags" -msgstr "Etiquetas Permitidas" +msgstr "Etiquetas permitidas" #: cps/templates/user_table.html:140 -#, fuzzy msgid "Edit Denied Tags" -msgstr "Seleccionar Etiquetas Denegadas" +msgstr "Editar etiquetas prohibidas" #: cps/templates/user_table.html:140 msgid "Denied Tags" -msgstr "Etiquetas Denegadas" +msgstr "Etiquetas prohibidas" #: cps/templates/user_table.html:141 -#, fuzzy msgid "Edit Allowed Column Values" msgstr "Editar valores permitidos para la columna" #: cps/templates/user_table.html:141 -#, fuzzy msgid "Allowed Column Values" msgstr "Valores permitidos de la columna" #: cps/templates/user_table.html:142 -#, fuzzy msgid "Edit Denied Column Values" -msgstr "Editar valores no permitidos para la columna" +msgstr "Editar valores prohibidos para la columna" #: cps/templates/user_table.html:142 -#, fuzzy msgid "Denied Column Values" -msgstr "Valores no permitidos de la columna" +msgstr "Valores prohibidos de la columna" #: cps/templates/user_table.html:144 -#, fuzzy msgid "Change Password" msgstr "Cambiar la contraseña" @@ -3609,17 +3605,14 @@ msgid "View" msgstr "Vista" #: cps/templates/user_table.html:150 -#, fuzzy msgid "Edit Public Shelves" -msgstr "Editar Estanterías públicas" +msgstr "Editar estanterías públicas" #: cps/templates/user_table.html:152 -#, fuzzy msgid "Sync selected Shelves with Kobo" -msgstr "sincronizar estanterías seleccionadas con Kobo" +msgstr "Sincronizar estanterías seleccionadas con Kobo" #: cps/templates/user_table.html:156 -#, fuzzy msgid "Show Read/Unread Section" -msgstr "Mostrar selección leidos/no leidos" +msgstr "Mostrar sección de leidos/no leidos" diff --git a/cps/translations/fi/LC_MESSAGES/messages.mo b/cps/translations/fi/LC_MESSAGES/messages.mo index 4eea2624..f2321df5 100644 Binary files a/cps/translations/fi/LC_MESSAGES/messages.mo and b/cps/translations/fi/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/fi/LC_MESSAGES/messages.po b/cps/translations/fi/LC_MESSAGES/messages.po index a14dfeda..f794ee52 100644 --- a/cps/translations/fi/LC_MESSAGES/messages.po +++ b/cps/translations/fi/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2020-01-12 13:56+0100\n" "Last-Translator: Samuli Valavuo \n" "Language: fi\n" @@ -16,509 +16,527 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Tilastot" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Palvelin uudelleenkäynnistetty, ole hyvä ja päivitä sivu" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Palvelinta sammutetaan, ole hyvä ja sulje sivu" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Kirja lisätty onnistuneeksi lähetettäväksi osoitteeseen %(eReadermail)s" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Tuntematon" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Ylläpitosivu" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Perusasetukset" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Käyttöliittymän asetukset" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Pääkäyttäjä" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Kaikki" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Näytä kaikki" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web asetukset päivitetty" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Oletko varma, että haluat poistaa hyllyn?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Oletko varma, että haluat poistaa hyllyn?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Oletko varma, että haluat poistaa hyllyn?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Oletko varma, että haluat poistaa hyllyn?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Haluatko varmasti pysäyttää Calibre-Webin?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Muuta SMTP asetuksia" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Testisähköpostin lähetyksessä tapahtui virhe: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Sähköpostipalvelimen tiedot päivitetty" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Tapahtui tuntematon virhe. Yritä myöhemmin uudelleen." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Muokkaa käyttäjää %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Käyttäjän %(user)s salasana palautettu" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Ole hyvä ja aseta SMTP postiasetukset ensin..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Lokitiedoston katselin" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Haetaan päivitystiedostoa" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Ladataan päivitystiedostoa" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Puretaan päivitystiedostoa" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Korvataan tiedostoja" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Tietokantayhteydet on katkaistu" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Sammutetaan palvelin" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Päivitys valmistui, ole hyvä ja paina OK ja lataa sivu uudelleen" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Päivitys epäonnistui:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP virhe" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Yhteysvirhe" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Aikakatkaisu yhteyttä luotaessa" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Yleinen virhe" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Sähköpostipalvelimen tiedot päivitetty" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Ominaisuuksien asetukset" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Ole hyvä ja täytä kaikki kentät!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Sähköpostiosoite ei ole toimivasta domainista" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Lisää uusi käyttäjä" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Käyttäjä '%(user)s' lisätty" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Tälle sähköpostiosoitteelle tai tunnukselle löytyi jo tili." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Käyttäjä '%(nick)s' poistettu" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Pääkäyttäjiä ei jää jäljelle, käyttäjää ei voi poistaa" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Käyttäjä '%(nick)s' päivitetty" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Hae" + #: cps/converter.py:31 msgid "not installed" msgstr "ei asennettu" @@ -527,128 +545,123 @@ msgstr "ei asennettu" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Ei mitään" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Virhe eKirjan avaamisessa. Tiedostoa ei ole tai se ei ole saatavilla:" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadata päivitetty onnistuneesti" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Tiedosto %(file)s tallennettu" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Lähteen tai kohteen tiedostomuoto puuttuu" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Kirja lisätty muutosjonoon muotoon %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Kirjan muunnoksessa tapahtui virhe: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Virhe eKirjan avaamisessa. Tiedostoa ei ole tai se ei ole saatavilla:" + +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s ei ole kelvollinen kieli" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadata päivitetty onnistuneesti" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Tiedostopääte '%(ext)s' ei ole sallittujen palvelimelle ladattavien listalla" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Tiedostopääte '%(ext)s' ei ole sallittujen palvelimelle ladattavien listalla" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Ladattavalla tiedostolla on oltava tiedostopääte" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "muokkaa metadataa" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Polun %(path)s luonti epäonnistui (Ei oikeutta)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Tiedoston %(file)s tallennus epäonnistui." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Tiedostoformaatti %(ext)s lisätty %(book)s" @@ -661,482 +674,475 @@ msgstr "Google Drive asetukset ei ole valmiit. Koita poistaa Google Drive käyt msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Paluuosoitteen domain ei ole varmistettu, seuraa ohjeita vamistaaksesi sen googlen kehittäjäkonsolissa" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s tiedostomuotoa ei löytynyt kirjalle: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s ei löytynyt Google Drivesta: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s ei löydy: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Lähetä Kindleen" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Tämä sähköposti on lähetetty Calibre-Web:sta." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web testisähköposti" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Testi sähköposti" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Aloita Calibre-Web:in käyttö" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Rekiströintisähköposti käyttäjälle: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Muunna %(orig)s muotoon %(format)s ja lähetä Kindleen" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Lähetä %(format)s Kindleen" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Lähetä Kindleen" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Haettua tiedostoa ei pystytty lukemaan. Ehkä vaäärät oikeudet?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Tiedon muuttaminen arvosta: '%(src)s' arvoon '%(dest)s' epäonnistui virheeseen: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Tiedostoa %(file)s ei löytynyt Google Drivesta" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Tiedon muuttaminen arvosta: '%(src)s' arvoon '%(dest)s' epäonnistui virheeseen: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Kirjan polkua %(path)s ei löytynyt Google Drivesta" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Löydä" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Rekisteröi tuottajalle %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "olet nyt kirjautunut tunnuksella: \"%(nickname)s\"" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "GitHubiin kirjautuminen epäonnistui." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Käyttäjätietojen haku GitHubista epäonnistui" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Googleen kirjautuminen epäonnistui." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Käyttäjätietojen haku Googlesta epäonnistui." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth virhe, yritä myöhemmin uudelleen." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth virhe, yritä myöhemmin uudelleen." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Kirjaudu sisään" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Valtuutusta ei löytynyt" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Valtuutus vanhentunut" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Onnistui! Ole hyvä ja palaa laitteellesi" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Kirjat" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Näytä viimeisimmät kirjat" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Kuumat kirjat" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Näytä kuumat kirjat" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Parhaiten arvioidut kirjat" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Näytä parhaiten arvioidut kirjat" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Luetut kirjat" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Näytä luetut ja lukemattomat" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Lukemattomat kirjat" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Näyt lukemattomat" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Löydä" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Näytä satunnausia kirjoja" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategoriat" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Näytä kategoriavalinta" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Sarjat" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Näytä sarjavalinta" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Kirjailijat" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Näytä kirjailijavalinta" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Julkaisijat" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Näytä julkaisijavalinta" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Kielet" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Näytä keilivalinta" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Arvostelut" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Näytä arvosteluvalinta" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Tiedotomuodot" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Näytä tiedostomuotovalinta" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Näytä viimeisimmät kirjat" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Hae" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Julkaistu alkaen " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Julkaisut ennen " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Arvostelu <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Arvostelu >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Edistynyt haku" @@ -1192,7 +1198,7 @@ msgstr "Kirja on poistettu hyllystä: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "luo hylly" @@ -1247,45 +1253,45 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Hylly: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Virhe hyllyn avauksessa. Hyllyä ei ole tai se ei ole saatavilla" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tehtävät" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Odottaa" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Epäonnistui" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Aloitettu" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Valmistui" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Tuntematon tila" @@ -1318,176 +1324,176 @@ msgstr "Uusi päivitys saatavilla. Paina alla olevaa nappia päivittääksesi ve msgid "No release information available" msgstr "Ei päivitystietoa saatavilla" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Löydä (satunnaiset kirjat)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Kuumat kirjat (ladatuimmat)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Kirjailija: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Julkaisija: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Sarja: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Arvostelu: %(rating)s tähteä" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Tiedostomuoto: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategoria: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Kieli: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "DLS" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Arvostelulistaus" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Tiedostomuotolistaus" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Ole hyvä ja aseta SMTP postiasetukset ensin..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Kirja lisätty onnistuneeksi lähetettäväksi osoitteeseen %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Kirjan: %(res)s lähettämisessa tapahtui virhe" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Ole hyvä ja aseta Kindle sähköpostiosoite ensin..." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Rekisteröi" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Sähköpostiosoitteellasi ei ole sallittua rekisteröityä" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Vahvistusviesti on lähetetty sähköpostiosoitteeseesi." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "LDAP autnetikoinnin aktivointi ei onnistu" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "olet nyt kirjautunut tunnuksella: \"%(nickname)s\"" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Väärä käyttäjätunnus tai salasana" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Tapahtui tuntematon virhe. Yritä myöhemmin uudelleen." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Väärä käyttäjätunnus tai salasana" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "olet nyt kirjautunut tunnuksella: \"%(nickname)s\"" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)sn profiili" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profiili päivitetty" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Tälle sähköpostiosoitteelle läytyi jo käyttäjätunnus." @@ -1496,54 +1502,58 @@ msgstr "Tälle sähköpostiosoitteelle läytyi jo käyttäjätunnus." msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Lähetä Kindleen" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "E-kirjan muunnos epäonnistui: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "E-kirjan muunnos epäonnistui: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1552,30 +1562,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "muokkaa metadataa" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Lähetä" @@ -1594,12 +1600,12 @@ msgstr "Lempinimi" msgid "Email" msgstr "Sähköposti" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Ylläpito" @@ -1609,8 +1615,8 @@ msgstr "Ylläpito" msgid "Password" msgstr "Salasana" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Lataa" @@ -1821,13 +1827,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "" @@ -1877,16 +1883,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "vähennä" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Enemmän" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Kieli" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Julkaisija" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Kuvaus:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Edellinen" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Seuraava" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Koti" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Kirjaudu ulos" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Poista kirja" @@ -1915,99 +1981,107 @@ msgstr "Muunna muotoon:" msgid "Convert book" msgstr "Muunna kirja" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Ladataan..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Sulje" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Virhe" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Lataus tehty, prosessoidaan, ole hyvä ja odota..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Lataa tiedostomuoto" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Kirjan otsikko" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Kirjailija" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Tunnisteet" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Julkaisupäivä" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Arvostelu" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Kuvaus" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Tunnisteet" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Arvostelu" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Kannen osoite (jpg, kuva ladataan ja tallennetaan tietokantaan, kenttä jää uudelleen tyhjäksi)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Lataa kuva paikalliselta levyltä" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Julkaisupäivä" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Julkaisija" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Kieli" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Kyllä" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Ei" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Lataa tiedostomuoto" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "katso kirjaa muokkauksen jälkeen" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Hae metadata" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2015,38 +2089,32 @@ msgstr "Hae metadata" msgid "Save" msgstr "" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Avainsana" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Hae avainsanaa " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Klikkaa kantta ladataksesi metadata lomakkeelle" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Ladataan..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Sulje" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Lähde" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Hakuvirhe!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Ei osumia! Kokeile jotain tosita hakusanaa." @@ -2155,7 +2223,7 @@ msgid "Enter " msgstr "Rekisteröi" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Oletko aivan varma?" @@ -2632,74 +2700,61 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Lue selaimessa" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Kuuntele selaimessa" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Merkitse lukemattomaksi" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Merkitse luetuksi" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Merkitse lukemattomaksi" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Luettu" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Hae" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Kuvaus:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Lisää hyllyyn" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Muokkaa metadataa" @@ -2771,10 +2826,6 @@ msgstr "Syötä domainnimi" msgid "Denied Domains (Blacklist)" msgstr "" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Seuraava" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2795,11 +2846,16 @@ msgstr "" msgid "Create Issue" msgstr "Luo virheilmoitus" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Ominaisuuksien asetukset" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Palaa kotiin" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2889,7 +2945,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2898,60 +2954,36 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Koti" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Vaihda navigointi" -#: cps/templates/layout.html:47 -msgid "Search Library" +#: cps/templates/layout.html:59 +msgid "Simple Theme" msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Tili" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Kirjaudu ulos" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Ladataan..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Virhe" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Lataus tehty, prosessoidaan, ole hyvä ja odota..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Asetukset" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Selaa" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Tietoja" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Edellinen" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Kirjan tiedot" @@ -3067,7 +3099,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Kirja" @@ -3076,36 +3108,81 @@ msgstr "Kirja" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web e-kirjaluettelo" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "PDF lukija" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Valitse käyttäjänimi" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Vaalea" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Tumma" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Palaa" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Uudelleenjärjestä teksti kun sivut on auki." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Poista" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Odottaa" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Pystysuunta" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Luettu" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3287,10 +3364,6 @@ msgstr "Linkki vanhenee 10 minuutissa." msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "" @@ -3307,11 +3380,11 @@ msgstr "Julkaisupäivästä" msgid "Published Date To" msgstr "Julkaisupäivään" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3348,11 +3421,13 @@ msgstr "Arvio enemmän kun" msgid "Rating Below" msgstr "Arvio vähemmän kun" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3376,6 +3451,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "pitäisikö hyllyn olla julkinen?" @@ -3444,15 +3527,19 @@ msgstr "Edistyminen" msgid "Run Time" msgstr "Ajoaika" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3460,6 +3547,10 @@ msgstr "" msgid "Reset user Password" msgstr "Nollaa käyttäjän salasana" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Näytä kirjat kielellä" diff --git a/cps/translations/fr/LC_MESSAGES/messages.mo b/cps/translations/fr/LC_MESSAGES/messages.mo index 91be39e6..f0cf43bc 100644 Binary files a/cps/translations/fr/LC_MESSAGES/messages.mo and b/cps/translations/fr/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/fr/LC_MESSAGES/messages.po b/cps/translations/fr/LC_MESSAGES/messages.po index 5f9de4a4..51221ece 100644 --- a/cps/translations/fr/LC_MESSAGES/messages.po +++ b/cps/translations/fr/LC_MESSAGES/messages.po @@ -22,7 +22,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2020-06-07 06:47+0200\n" "Last-Translator: \n" "Language: fr\n" @@ -31,512 +31,530 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistiques" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Serveur redémarré, merci de rafraîchir la page" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Arrêt du serveur en cours, merci de fermer la fenêtre" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Commande inconnue" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Teste les courriels en file d’attente pour l’envoi à %(email)s, veuillez vérifier le résultat des tâches" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Inconnu" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Page admin" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Configuration principale" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Configuration de l’interface utilisateur" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "La colonne personnalisée No.%(column)d n'existe pas dans la base de données calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Éditer les utilisateurs" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Tout" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "L'utilisateur n'a pas été trouvé" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} utilisateurs supprimés avec succès" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Montrer tout" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Demande malformée" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Le nom de l’invité ne peut pas être modifié" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "L’invité ne peut pas avoir ce rôle" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Aucun utilisateur admin restant, impossible de supprimer le rôle admin" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "La valeur doit être vraie ou fausse" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Rôle invalide" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "L’invité ne peut pas avoir cette vue" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Vue invalide" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Les paramètres régionaux de l’invité sont déterminés automatiquement et ne peuvent pas être définis" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Aucun paramètre régional valide n’est donné" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Aucune langue de livre valide donnée" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Paramètre non trouvé" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Colonne de lecture non valide" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Colonne restreinte non valide" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Configuration de Calibre-Web mise à jour" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Voulez-vous vraiment supprimer le jeton Kobo ?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Voulez-vous vraiment supprimer ce domaine ?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Voulez-vous vraiment supprimer cet utilisateur ?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Voulez-vous vraiment supprimer l’étagère ?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Voulez-vous vraiment supprimer l’étagère ?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Voulez-vous vraiment modifier les langues de livre visibles pour le ou les utilisateurs sélectionnés ?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Voulez-vous vraiment modifier le rôle sélectionné pour le ou les utilisateurs sélectionnés ?" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Voulez-vous vraiment modifier les restrictions sélectionnées pour le ou les utilisateurs sélectionnés ?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Voulez-vous vraiment modifier les restrictions de visibilité sélectionnées pour le ou les utilisateurs sélectionnés ?" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Voulez-vous vraiment supprimer l’étagère?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Voulez-vous vraiment arrêter Calibre-Web ?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Êtes-vous certain de vouloir supprimer la base de données de synchronisation de Calibre-Web pour forcer une synchronisation complète avec votre liseuse Kobo ?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Refuser" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Autoriser" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} entrées de synchronisation supprimées" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Étiquette introuvable" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Action invalide" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json n'est pas configuré pour l'application Web" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "L'emplacement du fichier logfile est incorrect, veuillez saisir un chemin valide" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "L'emplacement du fichier Access Logfile est incorrect, veuillez saisir un chemin valide" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Veuillez saisir un fournisseur LDAP, Port, DN et l'identifiant objet de l'utilisateur" -#: cps/admin.py:1211 +#: cps/admin.py:1223 #, fuzzy msgid "Please Enter a LDAP Service Account and Password" msgstr "Veuillez entrer un nom d'utilisateur valide pour réinitialiser le mot de passe" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Veuillez entrer un compte de service LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Le filtre objet du groupe LDAP a besoin d'un identifiant de format \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Le filtre objet du groupe LDAP a une parenthèse non gérée" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Le filtre objet de l'utilisateur LDAP a besoin d'un identifiant de format \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Le filtre objet de l'utilisateur LDAP a une parenthèse non gérée" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Le filtre utilisateur des membres LDAP doit avoir un identificateur de format \"%s\\ »" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Le filtre utilisateur de membre LDAP a des parenthèses non appariées" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CACertificat, certificat ou emplacement de clé non valide, veuillez entrer le chemin correct" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Ajouter un nouvel utilisateur" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Modifier les paramètres du serveur de courriels" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Erreur de la base de données: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Teste les courriels en file d’attente pour l’envoi à %(email)s, veuillez vérifier le résultat des tâches" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Il y a eu une erreur pendant l’envoi du courriel de test : %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Veuillez d'abord configurer votre adresse de courriel..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Les paramètres du serveur de courriels ont été mis à jour" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Une erreur inconnue est survenue. Veuillez réessayer plus tard." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Éditer l'utilisateur %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Le mot de passe de l’utilisateur %(user)s a été réinitialisé" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Veuillez configurer les paramètres SMTP au préalable..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Visualiseur de fichier journal" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Demande de mise à jour" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Téléchargement de la mise à jour" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Décompression de la mise à jour" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Remplacement des fichiers" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Les connexions à la base de données ont été fermées" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Arrêt du serveur" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Mise à jour terminée, merci d’appuyer sur okay et de rafraîchir la page" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "La mise à jour a échoué :" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Erreur HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Erreur de connexion" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Délai d'attente dépassé lors de l'établissement de connexion" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Erreur générale" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Le fichier de mise à jour ne peut pas être sauvegardé dans le répertoire temporaire" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Les fichiers n’ont pas pu être remplacés pendant la mise à jour" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Impossible de créer au moins un utilisateur LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Impossible de créer au moins un utilisateur LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Erreur : %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Erreur : Aucun utilisateur renvoyé dans la réponse LDAP du serveur" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Au moins un utilisateur LDAP n'a pas été trouvé dans la base de données" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} utilisateur importé avec succès" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "L'emplacement de la base de données est incorrect, veuillez saisir un chemin valide" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "La base de données n'est pas accessible en écriture" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "L'emplacement du fichier Keyfile est incorrect, veuillez saisir un chemin valide" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "L'emplacement du fichier Certfile est incorrect, veuillez saisir un chemin valide" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Les paramètres du serveur de courriels ont été mis à jour" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Configuration des options" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Veuillez compléter tous les champs !" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Cette adresse de courriel n’appartient pas à un domaine valide" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Ajouter un nouvel utilisateur" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Utilisateur '%(user)s' créé" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Un compte existant a été trouvé pour cette adresse de courriel ou pour ce surnom." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Utilisateur '%(nick)s' supprimé" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Impossible de supprimer l’utilisateur Invité" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Aucun utilisateur admin restant, impossible de supprimer l’utilisateur" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Utilisateur '%(nick)s' mis à jour" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Chercher" + #: cps/converter.py:31 msgid "not installed" msgstr "non installé" @@ -545,128 +563,123 @@ msgstr "non installé" msgid "Execution permissions missing" msgstr "Les permissions d'exécutions manquantes" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "La colonne personnalisée No.%(column)d n'existe pas dans la base de données calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Aucun" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Erreur d'ouverture du livre numérique. Le fichier n'existe pas ou n'est pas accessible" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Les identificateurs ne sont pas sensibles à la casse, écrasant l’ancien identificateur" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Les métadonnées ont bien été mises à jour" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Le fichier %(file)s a été téléchargé" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Le format de conversion de la source ou de la destination est manquant" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Le livre a été mis avec succès en file de traitement pour conversion vers %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Une erreur est survenue au cours de la conversion du livre : %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Le fichier téléchargé existe probablement dans la librairie, veuillez le modifier avant de le télécharger de nouveau: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Erreur d'ouverture du livre numérique. Le fichier n'existe pas ou n'est pas accessible" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Les identificateurs ne sont pas sensibles à la casse, écrasant l’ancien identificateur" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s n'est pas une langue valide" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Les métadonnées ont bien été mises à jour" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Le fichier téléchargé existe probablement dans la librairie, veuillez le modifier avant de le télécharger de nouveau: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "L’extension de fichier '%(ext)s' n’est pas autorisée pour être déposée sur ce serveur" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "L’extension de fichier '%(ext)s' n’est pas autorisée pour être déposée sur ce serveur" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Pour être déposé le fichier doit avoir une extension" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Le fichier %(filename)s ne peut pas être sauvegardé dans le répertoire temporaire" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Impossible de déplacer le fichier de couverture %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Le format du livre a été supprimé avec succès" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Le livre a été supprimé avec succès" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Vous n’avez par les permissions pour supprimer les livres" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "modifier les métadonnées" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s n’est pas un nombre valide, ignoré" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Impossible de créer le chemin %(path)s (Permission refusée)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Échec de la sauvegarde du fichier %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Le format de fichier %(ext)s a été ajouté à %(book)s" @@ -679,487 +692,480 @@ msgstr "La configuration de Google Drive n’est pas terminée, essayez de désa msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Le domaine de retour d’appel (Callback domain) est non vérifié, veuillez suivre les étapes nécessaires pour vérifier le domaine dans la console de développement de Google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "le format %(format)s est introuvable pour le livre : %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "le %(format)s est introuvable sur Google Drive : %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s introuvable : %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Envoyer vers Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Ce courriel a été envoyé depuis Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Courriel de test de Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Courriel de test" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Bien démarrer avec Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Courriel d’inscription pour l’utilisateur : %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Convertir de %(orig)s vers %(format)s et envoyer au Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Envoyer %(format)s vers le Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Envoyer vers Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Le fichier demandé n’a pu être lu. Problème de permission d’accès ?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Échec de la suppression du dossier de livre pour le livre %(id)s, le chemin d’accès comporte des sous-dossiers : %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "La suppression du livre %(id)s a échoué: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, fuzzy, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Suppression du livre %(id)s, le chemin du livre est invalide : %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Renommer le titre de : '%(src)s' à '%(dest)s' a échoué avec l’erreur : %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Le fichier %(file)s n'a pas été trouvé dans Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Renommer le titre de : '%(src)s' à '%(dest)s' a échoué avec l’erreur : %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Le chemin du livre %(path)s n'a pas été trouvé dans Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Cet utilisateur est déjà pris" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Format de l’adresse courriel invalide" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Erreur lors du téléchargement de la couverture" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Erreur de format de couverture" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Impossible de créer le chemin pour la couverture" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Le fichier couverture n'est pas un fichier image valide, ou ne peut pas être stocké" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Seuls les fichiers jpg/jpeg/png/webp/bmp sont supportés comme fichier de couverture" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Contenu du fichier de couverture invalide" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Seuls les fichiers jpg/jpeg sont supportés comme fichier de couverture" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Découvrir" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Fichier binaire UnRar non trouvé" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Une erreur est survenue lors de l'exécution d'UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "La base de données n'est pas accessible en écriture" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Les permissions d'exécutions manquantes" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Une erreur est survenue lors de l'exécution d'UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Veuilllez ne pas accéder à calibre-web par localhost pour obtenir un api_endpoint valide pour un appareil kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Configuration Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Enregistrer avec %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "vous êtes maintenant connecté comme : '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Lien vers %(oauth)s effectué avec succès" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "La connexion a échoué, aucun utilisateur lié au compte OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Suppression de la liaison vers %(oauth)s effectuée avec succès" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Suppression de la liaison vers %(oauth)s a échoué" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Non lié à %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Échec de la connexion avec GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Impossible d’obtenir les informations d’utilisateur à partir de GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Échec de la connexion avec Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Impossible d’obtenir les informations d’utilisateur avec Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Erreur Oauth GitHub, veuillez réessayer plus tard." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Erreur Oauth Github : {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Erreur Oauth Google, veuillez réessayer plus tard." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Erreur Oauth Google : {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Étoiles" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Connexion" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Jeton non trouvé" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Jeton expiré" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Réussite! Merci de vous tourner vers votre appareil" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Livres" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Afficher les livres récents" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Livres populaires" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Montrer les livres populaires" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Livres téléchargés" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Montrer les livres téléchargés" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Livres les mieux notés" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Montrer les livres les mieux notés" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Livres lus" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Montrer lus et non-lus" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Livres non-lus" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Afficher non-lus" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Découvrir" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Montrer des livres au hasard" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Catégories" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Montrer la sélection par catégories" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Séries" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Montrer la sélection par séries" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Auteurs" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Montrer la sélection par auteur" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Éditeurs" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Montrer la sélection par éditeur" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Langues" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Montrer la sélection par langue" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Notes" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Afficher la sélection des évaluations" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formats de fichier" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Afficher la sélection des formats de fichiers" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Livres archivés" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Afficher les livres archivés" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Liste des livres" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Montrer la liste des livres" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Chercher" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Publié après le " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Publié avant le " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Évaluation <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Évaluation >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Status de lecture = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Erreur lors de la recherche de colonnes personnalisées, veuillez redémarrer Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Recherche avancée" @@ -1215,7 +1221,7 @@ msgstr "Le livre a été supprimé de l'étagère %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Désolé, vous n’êtes pas autorisé à supprimer un livre de cette étagère" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Créer une étagère" @@ -1270,45 +1276,45 @@ msgstr "Une étagère publique avec le nom '%(title)s' existe déjà." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Une étagère privée avec le nom '%(title)s' existe déjà." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Étagère : '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Erreur à l’ouverture de l’étagère. Elle n’existe plus ou n’est plus accessible" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tâches" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "En attente" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Echoué" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Débuté" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Terminé" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Statut inconnu" @@ -1341,178 +1347,178 @@ msgstr "Une nouvelle mise à jour est disponible. Cliquez sur le bouton ci-desso msgid "No release information available" msgstr "Aucune information concernant cette version n’est disponible" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Découvrir (Livres au hasard)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Livres populaires (les plus téléchargés)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Livres téléchargés par %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Auteur : %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Éditeur : '%(name)s'" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Séries : %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Évaluation : %(rating)s étoiles" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Format de fichier : %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Catégorie : %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Langue : %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Téléchargements" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Liste des évaluations" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Liste de formats de fichiers" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Veuillez configurer les paramètres SMTP au préalable..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Le livre a été mis en file de traitement avec succès pour un envoi vers %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Il y a eu une erreur en envoyant ce livre : %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Veuillez mettre à jour votre profil avec une adresse de courriel Kindle valide." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Créer un compte" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Le serveur de courriel n'est pas configuré, veuillez contacter votre administrateur!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Le serveur de courriel n'est pas configuré, veuillez contacter votre administrateur!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Votre adresse de courriel n’est pas autorisé pour une inscription" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Le courriel de confirmation a été envoyé à votre adresse." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Impossible d’activer l’authentification LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "vous êtes maintenant connecté comme : '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Connexion de secours comme: '%(nickname)s', le serveur LDAP est indisponible, ou l'utilisateur est inconnu" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Impossible de se connecter: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Mauvais nom d'utilisateur ou mot de passe" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Le nouveau mot de passe a été envoyé vers votre adresse de courriel" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Une erreur inconnue est survenue. Veuillez réessayer plus tard." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Veuillez entrer un nom d'utilisateur valide pour réinitialiser le mot de passe" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "vous êtes maintenant connecté comme : '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Profil de %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profil mis à jour" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Un compte existant a été trouvé pour cette adresse de courriel." @@ -1521,54 +1527,58 @@ msgstr "Un compte existant a été trouvé pour cette adresse de courriel." msgid "Found no valid gmail.json file with OAuth information" msgstr "Aucun fichier gmail.json avec information OAuth valide trouvé" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Envoyer vers Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "ebook-convert calibre %(tool)s non trouvé" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "Format %(format)s non trouvé sur le disque" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Le convertisseur Ebook a échoué avec une erreur inconnue" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "La commande Kepubify-converter a échouée : %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Fichier converti non trouvé ou plus d'un fichier dans le chemin %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "La commande ebook-convert a échouée : %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre a échoué avec l’erreur : %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "La commande ebook-convert a échouée : %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1577,30 +1587,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "modifier les métadonnées" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Téléverser" @@ -1619,12 +1625,12 @@ msgstr "Nom d'utilisateur" msgid "Email" msgstr "Adresse de courriel" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Envoyer vers une adresse de courriel Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Administration" @@ -1634,8 +1640,8 @@ msgstr "Administration" msgid "Password" msgstr "Mot de passe" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Télécharger" @@ -1847,13 +1853,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Annuler" @@ -1903,16 +1909,76 @@ msgstr "Trier en fonction de la date de publication, le plus récent en premier" msgid "Sort according to publishing date, oldest first" msgstr "Trier en fonction de la date de publication, le plus ancien en premier" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "réduire" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Plus de" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Livre %(index)s sur %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Langue" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Éditeur" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publié" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Description :" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Précédent" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Suivant" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Aucun résultat trouvé" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Accueil" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Chercher dans librairie" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Déconnexion" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Supprimer le livre" @@ -1941,99 +2007,107 @@ msgstr "Convertir vers :" msgid "Convert book" msgstr "Convertir le livre" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Téléversement en cours..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Fermer" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Erreur" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Téléversement terminé, traitement en cours, veuillez patienter…." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Format du fichier téléversé" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Titre du livre" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Auteur" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Étiquettes" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID de séries" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Date de publication" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Évaluation" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Description" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identifiants" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Type d'identifiant" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Valeur d'identifiant" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Supprimer" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Ajouter un identifiant" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Étiquettes" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID de séries" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Évaluation" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Obtenir la couverture à partir d'une URL (JPEG - l'image sera téléchargée et sauvegardée dans la base de données)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Téléverser la couverture depuis un fichier en local" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Date de publication" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Éditeur" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Langue" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Oui" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Non" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Format du fichier téléversé" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Voir le livre lors de la sauvegarde" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Obtenir les métadonnées" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2041,38 +2115,32 @@ msgstr "Obtenir les métadonnées" msgid "Save" msgstr "Sauvegarder" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Mot-clé" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Rechercher le mot-clé " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Cliquer sur la couverture pour importer les métadonnées dans le formulaire" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Chargement..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Fermer" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Source" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Erreur lors de la recherche!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Aucun résultat. Veuillez essayer avec un nouveau mot clé." @@ -2181,7 +2249,7 @@ msgid "Enter " msgstr "Identifiants" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Êtes-vous vraiment sûr?" @@ -2660,74 +2728,61 @@ msgstr "Ajouter les étiquettes autorisées/refusées" msgid "Add Allowed/Denied custom column values" msgstr "Ajouter les valeurs de colonnes autorisées/refusées" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Lire dans le navigateur" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Écouter dans le navigateur" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Livre %(index)s sur %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publié" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Marquer comme non lu" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Marquer comme lu" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Marquer comme non lu" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Lu" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Restaurer à partir de l'archive" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Ajouter comme archive" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Archivé" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Description :" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Ajouter à l'étagère" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Public)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Éditer les métadonnées" @@ -2800,10 +2855,6 @@ msgstr "Saisir le nom du domaine" msgid "Denied Domains (Blacklist)" msgstr "Domaines refusés (Liste noire)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Suivant" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Ouvrir le fichier .kobo/Kobo/Kobo eReader.conf dans un éditeur de texte et ajouter (ou éditer):" @@ -2826,11 +2877,16 @@ msgstr "Le serveur de courriel n'est pas configuré, veuillez contacter votre ad msgid "Create Issue" msgstr "Signaler un problème" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Configuration des options" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Retour à l’accueil" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Déconnecter l’utilisateur" @@ -2920,7 +2976,7 @@ msgstr "Livres classés par évaluation" msgid "Books ordered by file formats" msgstr "Livres classés par formats de fichiers" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Etagères" @@ -2929,60 +2985,37 @@ msgstr "Etagères" msgid "Books organized in shelves" msgstr "Livres organisés par étagères" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Accueil" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Basculer la navigation" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Chercher dans librairie" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Simple" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Compte" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Déconnexion" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Téléversement en cours..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Erreur" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Téléversement terminé, traitement en cours, veuillez patienter…." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Paramètres" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Veuillez ne pas rafraîchir la page" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Explorer" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "À propos" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Précédent" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Détails du livre" @@ -3098,7 +3131,7 @@ msgstr "Répertoire parent" msgid "Select" msgstr "Sélectionner" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Livre" @@ -3107,36 +3140,82 @@ msgstr "Livre" msgid "Calibre-Web eBook Catalog" msgstr "Catalogue de livres électroniques Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "Lecteur PDF" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Choisissez un nom d'utilisateur" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Clair" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Sombre" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Retour" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Mettre à jour la mise en page du texte quand les bandeaux latéraux sont ouverts." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Supprimer" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "En attente" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Vertical" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Lu" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Colonne de lecture non valide" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3318,10 +3397,6 @@ msgstr "Le lien expirera après 10 minutes." msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Aucun résultat trouvé" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Chercher le terme:" @@ -3338,11 +3413,11 @@ msgstr "Date de publication (depuis)" msgid "Published Date To" msgstr "Date de publication (jusqu’à)" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3379,11 +3454,13 @@ msgstr "Évaluation supérieure à" msgid "Rating Below" msgstr "Évaluation inférieure à" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Depuis" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Vers" @@ -3407,6 +3484,16 @@ msgstr "Désactiver l’ordre de modification" msgid "Enable Change order" msgstr "Activer l’ordre de modification" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Trier en fonction de la date du livre, le plus récent en premier" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Trier en fonction de la date du livre, le plus ancien en premier" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Partager avec tout le monde" @@ -3475,15 +3562,20 @@ msgstr "Avancement" msgid "Run Time" msgstr "Durée" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Fusionner" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3491,6 +3583,10 @@ msgstr "" msgid "Reset user Password" msgstr "Réinitialiser le mot de passe de l’utilisateur" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Montrer les livres dans la langue" diff --git a/cps/translations/gl/LC_MESSAGES/messages.mo b/cps/translations/gl/LC_MESSAGES/messages.mo index 0fb693e0..4b8dd0da 100644 Binary files a/cps/translations/gl/LC_MESSAGES/messages.mo and b/cps/translations/gl/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/gl/LC_MESSAGES/messages.po b/cps/translations/gl/LC_MESSAGES/messages.po index 6c036453..f08e1f16 100644 --- a/cps/translations/gl/LC_MESSAGES/messages.po +++ b/cps/translations/gl/LC_MESSAGES/messages.po @@ -5,510 +5,524 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" -"PO-Revision-Date: 2022-08-11 16:46+0200\n" -"Last-Translator: pollitor \n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-12-08 13:50+0100\n" +"Last-Translator: pollitor@gmx.com\n" "Language: gl\n" "Language-Team: gl\n" "Plural-Forms: nplurals=2; plural=(n != 1);\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Estatísticas" -#: cps/admin.py:150 -#, fuzzy +#: cps/admin.py:151 msgid "Server restarted, please reload page." -msgstr "Servidor reiniciado. Por favor, recargue a páxina" +msgstr "Servidor reiniciado. Por favor, recargue a páxina." -#: cps/admin.py:152 -#, fuzzy +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." -msgstr "O servidor estase apagando. Por favor, peche a xanela" +msgstr "O servidor estase apagando, por favor peche a xanela." -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" -msgstr "" +msgstr "Éxito! Base de datos reconectada" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Orde descoñecida" -#: cps/admin.py:174 -#, fuzzy +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" -msgstr "Posto en cola un correo electrónico de proba enviado a %(email)s, por favor, comproba o resultado nas Tarefas" +msgstr "Éxito! Libros postos na cola para a copia de seguridade dos metadatos, por favor comproba o resultado nas Tarefas" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Descoñecido" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Páxina de administración" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Configuración Básica" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Configuración da Interface de Usuario" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Columna personalizada Num. %(column)d non existe na base de datos calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Editar Usuarios" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Todo" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Usuario non atopado" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} usuarios borrados con éxito" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Mostrar Todo" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Petición mal formada" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" -msgstr "O nome do convidado non se pode cambiar" +msgstr "Non se pode cambiar o nome do convidado" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" -msgstr "O convidado non pode ter este rol" +msgstr "O convidado non pode ter este papel" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" -msgstr "Non queda ningún usuario administrador, non se pode eliminar ao usuario" +msgstr "Non queda ningún usuario administrador, non se pode eliminar o rol de administrador" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "O Valor ten que ser verdadeiro ou falso" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Rol non válido" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "O convidado non pode ter esta vista" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Vista non válida" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" -msgstr "O sitio do convidado determínase automáticamente e non se pode cambiar" +msgstr "A localización do convidado determínase automáticamente e non se pode cambiar" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Non hai unha localización válida" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" -msgstr "Non se indicou unha lingua válida para o libro" +msgstr "Non se indica ningún idioma de libro válido" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" -msgstr "Parámetro non atopado" +msgstr "Non se atopou o parámetro" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Columna de lectura non válida" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Columna restrinxida non válida" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" -msgstr "Configuración de Calibre-Web actualizada" +msgstr "Actualizouse a configuración de Calibre-Web" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" -msgstr "De verdade queres borrar o Token de Kobo?" +msgstr "De verdade queres eliminar o token Kobo?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" -msgstr "De verdade desexas borrar este dominio?" +msgstr "De verdade queres eliminar este dominio?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "De verdade queres borrar este usuario?" -#: cps/admin.py:616 -msgid "Are you sure you want to delete this shelf?" -msgstr "De verdade queres eliminar este andel?" - -#: cps/admin.py:618 -msgid "Are you sure you want to change locales of selected user(s)?" -msgstr "De verdade queres cambiar a linguaxe dos usuarios seleccionados?" - -#: cps/admin.py:620 -msgid "Are you sure you want to change visible book languages for selected user(s)?" -msgstr "De verdade queres cambiar as linguas visibles do libro dos usuarios seleccionados?" - #: cps/admin.py:622 -msgid "Are you sure you want to change the selected role for the selected user(s)?" -msgstr "De verdade queres cambiar o rol seleccionado do usuario seleccionado?" +msgid "Are you sure you want to delete this shelf?" +msgstr "De verdade queres eliminar esta estantería?" #: cps/admin.py:624 -msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" -msgstr "De verdade queres cambiar as restricións escollidas dos usuarios seleccionados?" +msgid "Are you sure you want to change locales of selected user(s)?" +msgstr "Estás seguro de que queres cambiar a(s) rexión(s) local(/is) do(s) usuario(s) seleccionado(s)?" #: cps/admin.py:626 +msgid "Are you sure you want to change visible book languages for selected user(s)?" +msgstr "Estás seguro de que queres cambiar os idiomas dos libros visibles para o(s) usuario(s) seleccionado(s)?" + +#: cps/admin.py:628 +msgid "Are you sure you want to change the selected role for the selected user(s)?" +msgstr "Estás seguro de que queres cambiar o rol seleccionado para o(s) usuario(s) seleccionado(s)?" + +#: cps/admin.py:630 +msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" +msgstr "Estás seguro de que queres cambiar as restricións seleccionadas para o(s) usuario(s) seleccionado(s)?" + +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" -msgstr "De verdade queres cambiar as restricións de visibilidade dos usuarios seleccionados?" +msgstr "Estás seguro de que queres cambiar as restricións de visibilidade seleccionadas para o(s) usuario(s) seleccionado(s)?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" -msgstr "De verdade queres cambiar o comportamento da sincronización do andel para o usuario seleccionado?" +msgstr "Estás seguro de que queres cambiar o comportamento de sincronización da estantería para o(s) usuarios seleccionado(s)?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" -msgstr "De verdade queres cambiar a localización da biblioteca Calibre?" +msgstr "Estás seguro de que queres cambiar a localización da biblioteca de Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" -msgstr "Calibre-web buscará cubertas actualizadas e miniaturas de cubertas actualizadas, isto pode levar un intre?" +msgstr "Calibre-Web buscará portadas actualizadas e actualizará as miniaturas das portadas, isto pode levar un tempo?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" -msgstr "Está seguro que quere borrar a base de datos de sincronización de Calibre-Web para forzar unha sincronización completa co seu lector Kobo?" +msgstr "Estás seguro de que queres eliminar a base de datos de sincronización de Calibre-Web para forzar unha sincronización completa co teu Kobo Reader?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Denegar" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Permitir" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "Elimináronse {} entradas de sincronización" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" -msgstr "Etiqueta non atopada" +msgstr "Non se atopou a etiqueta" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Acción non válida" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json non está configurado para a aplicación web" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" -msgstr "A localización do arquivo de rexistro non é válida. Por favor, Introduce a ruta correcta" +msgstr "A localización do ficheiro de rexistro non é válida. Introduza o camiño correcto" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" -msgstr "A localización do rexistro de accesos non é válida. Por favor, Introduce a ruta correcta" +msgstr "A localización do ficheiro de rexistro de acceso non é válida. Introduza o camiño correcto" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" -msgstr "Por favor, Introduce un provedor LDAP, porto, DN e o User Object Identifier" +msgstr "Introduza un provedor LDAP, un porto, un DN e un identificador de obxecto de usuario" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" -msgstr "Por favor, introduce unha conta de servizo LDAP e o seu contrasinal" +msgstr "Introduza unha conta de servizo LDAP e un contrasinal" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" -msgstr "Por favor, introduce unha conta de servizo LDAP" +msgstr "Introduza unha conta de servizo LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" -msgstr "LDAP Group Object Filter necesita ter un identificador de formato \"%s\"" +msgstr "O filtro de obxectos de grupo LDAP debe ter un identificador de formato \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" -msgstr "O LDAP Group Object Filter ten parénteses que non casan" +msgstr "O filtro de obxectos de grupo LDAP ten parénteses sen coincidencia" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" -msgstr "LDAP Group Object Filter necesita ter un identificador de formato \"%s\"" +msgstr "O filtro de obxectos de usuario LDAP debe ter un identificador de formato \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" -msgstr "O LDAP Group Object Filter ten parénteses que non casan" +msgstr "O filtro de obxectos de usuario LDAP ten parénteses sen coincidencia" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" -msgstr "O filtro de usuarios LDAP necesita ter un identificador de formato \"%s\"" +msgstr "O filtro de usuario de membro LDAP debe ter un identificador de formato \"%s\"" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" -msgstr "O filtro de LDAP \"Member User\" ten parénteses que non casan" +msgstr "O filtro de usuarios de membros de LDAP ten parénteses sen coincidencia" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" -msgstr "As localizacións do certificado da CA do LDAP, do certificado ou da chave non válidos. Por favor introduce a ruta correcta" +msgstr "As localizacións do certificado da Autoridade de certificación (CA) do LDAP, do certificado ou da chave non son válidos. introduza o camiño correcto" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Engadir novo usuario" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" -msgstr "Cambiar os parámetros do correo" +msgstr "Editar a configuración do servidor de correo electrónico" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." -msgstr "" +msgstr "Éxito! Conta de Gmail verificada." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." -msgstr "Error na base de datos: %(error)s." +msgstr "Vaia! Erro da base de datos: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" -msgstr "Posto en cola un correo electrónico de proba enviado a %(email)s, por favor, comproba o resultado nas Tarefas" +msgstr "Un correo electrónico de proba púxose na cola para enviar a %(email)s, comprobe Tarefas para ver o resultado" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" -msgstr "Ocurreu un error enviando o correo electrónico de proba: %(res)s" +msgstr "Produciuse un erro ao enviar o correo electrónico de proba: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." -msgstr "Por favor, configure o seu correo electrónico primeiro..." +msgstr "Configure primeiro o seu enderezo de correo electrónico..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" -msgstr "Actualizáronse os axustes do servidor de correo electrónico" +msgstr "Actualizouse a configuración do servidor de correo electrónico" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" -msgstr "Editar a Configuración das Tarefas Programadas" +msgstr "Editar a configuración de tarefas programadas" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" -msgstr "Indicada unha hora incorrecta de comezo de tarefa" +msgstr "Hora de inicio non válida para a tarefa especificada" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" -msgstr "Indicada unha duracción incorrecta para a tarefa" +msgstr "Duración non válida para a tarefa especificada" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Actualizouse a configuración das tarefas programadas" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." -msgstr "Sucedeu un erro descoñecido. Por favor volva a intentalo máis tarde." +msgstr "Vaia! Produciuse un erro descoñecido. Téntao de novo máis tarde." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" -msgstr "A configuración da DB non se pode escribir" +msgstr "A base de datos de configuración non se pode escribir" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Editar o Usuario %(nick)s" -#: cps/admin.py:1445 -#, fuzzy, python-format +#: cps/admin.py:1457 +#, python-format msgid "Success! Password for user %(user)s reset" -msgstr "Reiniciada a contrasinal para o usuario %(user)s" +msgstr "Éxito! O contrasinal para o usuario %(user)s restableceuse" -#: cps/admin.py:1451 -#, fuzzy +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." -msgstr "Configura primeiro os parámetros do servidor SMTP..." +msgstr "Vaia! Configure a configuración do correo SMTP." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" -msgstr "Visor do ficheiro de rexistro" +msgstr "Visor de ficheiros de rexistro" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Solicitando paquete de actualización" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Descargando paquete de actualización" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" -msgstr "Descomprimendo paquete de actualización" +msgstr "Descomprimindo o paquete de actualización" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" -msgstr "Remplazando archivos" +msgstr "Substituindo ficheiros" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" -msgstr "As conexións coa base datos están pechadas" +msgstr "As conexións de base de datos están pechadas" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" -msgstr "Detendo o servidor" +msgstr "Parando o servidor" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" -msgstr "Actualización finalizada. Por favor, prema OK e recargue a páxina" +msgstr "A actualización finalizou, preme Aceptar e volve cargar a páxina" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" -msgstr "A actualización fallou:" +msgstr "Fallou a actualización:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Erro HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Erro de conexión" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" -msgstr "Tempo esgotado mentras se trataba de establecer a conexión" +msgstr "Tempo esgotado ao establecer a conexión" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Erro xeral" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" -msgstr "A actualización do arquivo non se puido gardar no directorio temporal (Temp Dir)" +msgstr "Non se puido gardar o ficheiro de actualización no directorio temporal" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Non se puideron substituír os ficheiros durante a actualización" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" -msgstr "Erro ao extraer polo menos un usuario LDAP" +msgstr "Produciuse un erro ao extraer polo menos un usuario LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" -msgstr "Erro ao crear polo menos un usuario LDAP" +msgstr "Produciuse un erro ao crear polo menos un usuario LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Erro: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" -msgstr "Erro: o servidor LDAP non devolveu ningún usuario" +msgstr "Erro: non se devolveu ningún usuario en resposta do servidor LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" -msgstr "Polo menos, un usuario LDAP non se atopou na base de datos" +msgstr "Non se atopou polo menos un usuario LDAP na base de datos" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" -msgstr "Usuario {} importado con éxito" +msgstr "Usuario {} importado correctamente" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "O camiño dos libros non é válido" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" -msgstr "A localización da base de datos non é válida. Por favor, Introduce a ruta correcta" +msgstr "A localización da base de datos non é válida, introduza o camiño correcto" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" -msgstr "A base de datos non é modificable" +msgstr "A base de datos non é escribible" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" -msgstr "A localización do Keyfile non é válida, por favor, Introduce a ruta correcta" +msgstr "A localización do ficheiro de chave non é válida. Introduza o camiño correcto" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" -msgstr "A localización do Certfile non é válida, por favor, Introduce a ruta correcta" +msgstr "A localización do ficheiro de certificado non é válida. Introduza o camiño correcto" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" -msgstr "" +msgstr "A lonxitude do contrasinal debe estar entre 1 e 40" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" -msgstr "Actualizados os axustes da base de datos" +msgstr "Actualizouse a configuración da base de datos" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Configuración da base de datos" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." -msgstr "Por favor, cubra todos os campos!" - -#: cps/admin.py:1926 -msgid "E-mail is not from valid domain" -msgstr "O correo electrónico non ven dun dominio válido" - -#: cps/admin.py:1932 -msgid "Add new user" -msgstr "Engadir un usuario novo" - -#: cps/admin.py:1943 -#, python-format -msgid "User '%(user)s' created" -msgstr "Usuario '%(user)s' creado" +msgstr "Vaia! Complete todos os campos." #: cps/admin.py:1949 -msgid "Oops! An account already exists for this Email. or name." -msgstr "Atopada unha conta existente para este correo electrónico ou nome de usuario." +msgid "E-mail is not from valid domain" +msgstr "O correo electrónico non é dun dominio válido" -#: cps/admin.py:1979 +#: cps/admin.py:1955 +msgid "Add new user" +msgstr "Engadir novo usuario" + +#: cps/admin.py:1966 +#, python-format +msgid "User '%(user)s' created" +msgstr "Creouse o usuario '%(user)s'" + +#: cps/admin.py:1972 +msgid "Oops! An account already exists for this Email. or name." +msgstr "Vaia! Xa existe unha conta para este correo electrónico ou nome." + +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" -msgstr "Usuario '%(nick)s' eliminado" +msgstr "Eliminouse o usuario '%(nick)s'" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" -msgstr "Non se pode borrar ao Usuario Invitado" +msgstr "Non se pode eliminar o usuario convidado" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" -msgstr "Non queda ningún usuario administrador, non se pode borrar ao usuario" +msgstr "Non queda ningún usuario administrador, non se pode eliminar o usuario" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" -msgstr "" +msgstr "O correo electrónico non pode estar baleiro e ten que ser un correo electrónico válido" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Usuario '%(nick)s' actualizado" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Busca" + #: cps/converter.py:31 msgid "not installed" msgstr "non instalado" @@ -517,1057 +531,1018 @@ msgstr "non instalado" msgid "Execution permissions missing" msgstr "Faltan permisos de execución" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Columna personalizada No.%(column)d non existe na base de datos calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Ningún" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "oh, oh, o libro seleccionado non está disponible. O arquivo non existe ou non está accesible" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "O usuario non ten permisos para subir a cuberta" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Os identificadores non distinguen entre maiúsculas e minúsculas, sobrescribindo o identificador antigo" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadatos actualizados con éxito" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Erro editando libro: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" -msgstr "O ficheiro %(file)s subiuse" +msgstr "Cargouse o ficheiro %(file)s" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" -msgstr "Falta a fonte ou o formato de destino para a conversión" +msgstr "Falta o formato de orixe ou destino para a conversión" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" -msgstr "Libro posto na cola para a súa conversión a %(book_format)s" +msgstr "O libro fixo cola correctamente para converterse a %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" -msgstr "Houbo un erro ao convertir este libro: %(res)s" +msgstr "Produciuse un erro ao converter este libro: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "O libro cargado probablemente existe na biblioteca, considera cambialo antes de subilo outra vez: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Vaia! O libro seleccionado non está dispoñible. O ficheiro non existe ou non é accesible" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "O usuario non ten dereitos para cargar portada" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Os identificadores non distinguen entre maiúsculas e minúsculas, sobreescribindo o identificador antigo" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" -msgstr "%(langname)s non é unha lingua válida" +msgstr "'%(langname)s' non é un idioma válido" -#: cps/editbooks.py:756 cps/editbooks.py:1192 -#, fuzzy +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Actualizáronse correctamente os metadatos" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Erro ao editar o libro: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Probablemente exista o libro cargado na biblioteca. Considera cambiar antes de cargar un novo: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" -msgstr "Non se permite subir arquivos coa extensión '%(ext)s' a este servidor" +msgstr "Non se permite cargar o tipo de ficheiro neste servidor" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" -msgstr "Non se permite subir arquivos coa extensión '%(ext)s' a este servidor" +msgstr "Non se permite cargar a extensión de ficheiro '%(ext)s' a este servidor" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" -msgstr "O arquivo que se vai cargar debe ter unha extensión" +msgstr "O ficheiro que se vai cargar debe ter unha extensión" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" -msgstr "El archivo %(filename)s non puido gravarse no directorio temporal (Temp Dir)" +msgstr "Non se puido gardar o ficheiro %(filename)s no directorio temporal" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" -msgstr "Fallo ao mover o arquivo de cuberta %(file)s: %(error)s" +msgstr "Produciuse un erro ao mover o ficheiro de portada %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" -msgstr "Formato de libro eliminado con éxito" +msgstr "O formato do libro eliminouse correctamente" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" -msgstr "Libro eliminado con éxito" +msgstr "Libro eliminado correctamente" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" -msgstr "Vostede non ten permisos para borrar libros" +msgstr "Faltan permisos para eliminar libros" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "editar metadatos" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" -msgstr "%(seriesindex)s non é un número válido, saltando" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "Seriesindex: %(seriesindex)s non é un número válido, saltando" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" -msgstr "O usuario non ten permisos para cargar formatos de ficheiro adicionais" +msgstr "O usuario non ten dereitos para cargar formatos de ficheiro adicionais" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." -msgstr "Fallo ao crear a ruta %(path)s (permiso denegado)" +msgstr "Produciuse un erro ao crear o camiño %(path)s (permiso denegado)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." -msgstr "Fallo ao gardar o arquivo %(file)s." +msgstr "Produciuse un erro ao almacenar o ficheiro %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" -msgstr "Arquivo con formato %(ext)s engadido a %(book)s" +msgstr "Engadiuse o formato de ficheiro %(ext)s a %(book)s" #: cps/gdrive.py:58 msgid "Google Drive setup not completed, try to deactivate and activate Google Drive again" -msgstr "A configuración de Google Drive non se completou, intente desactivar e activar Google Drive outra vez" +msgstr "Non se completou a configuración de Google Drive, tenta desactivalo e activalo de novo" #: cps/gdrive.py:96 msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" -msgstr "O dominio Callback non se comprobou, siga os pasos para comprobalo na consola de desenvolvedor de Google" +msgstr "O dominio de devolución de chamada non está verificado. Sigue os pasos para verificar o dominio na consola de desenvolvedores de Google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" -msgstr "%(format)s formato non atopado para o id do libro: %(book)d" +msgstr "Non se atopou o formato %(format)s para o ID do libro: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" -msgstr "%(format)s non atopado en Google Drive: %(fn)s" +msgstr "Non se atopou %(format)s en Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s non atopado: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" -msgstr "Enviar ao Kindle" +msgstr "Enviar a eReader" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 -#, fuzzy +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." -msgstr "Este correo electrónico enviouse empregando Calibre-Web." +msgstr "Este correo electrónico foi enviado a través de Calibre-Web." -#: cps/helper.py:120 -#, fuzzy +#: cps/helper.py:123 msgid "Calibre-Web Test Email" -msgstr "Correo de proba de Calibre-Web" +msgstr "Correo electrónico de proba de Calibre-Web" -#: cps/helper.py:121 -#, fuzzy +#: cps/helper.py:124 msgid "Test Email" -msgstr "Comprobar correo electrónico" +msgstr "Correo electrónico de proba" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" -msgstr "Primeiros pasos con Calibre-Web" +msgstr "Comeza con Calibre-Web" -#: cps/helper.py:143 -#, fuzzy, python-format +#: cps/helper.py:146 +#, python-format msgid "Registration Email for user: %(name)s" msgstr "Correo electrónico de rexistro para o usuario: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 -#, fuzzy, python-format +#: cps/helper.py:157 cps/helper.py:163 +#, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" -msgstr "Convertir %(orig)s a %(format)s e enviar ao Kindle" +msgstr "Converte %(orig)s en %(format)s e envía a eReader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 -#, fuzzy, python-format +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 +#, python-format msgid "Send %(format)s to eReader" -msgstr "Enviado %(format)s ao Kindle" +msgstr "Enviar %(format)s a eReader" -#: cps/helper.py:227 -#, fuzzy, python-format +#: cps/helper.py:230 +#, python-format msgid "%(book)s send to eReader" -msgstr "Enviar ao Kindle %(book)s" +msgstr "%(book)s enviado a eReader" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" -msgstr "O arquivo solicitado non pode lerse. Quizais existen problemas cos permisos?" +msgstr "Non se puido ler o ficheiro solicitado. Quizais permisos incorrectos?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" -msgstr "O estado de lectura non pode fixarse: {}" +msgstr "Non se puido establecer o estado de lectura: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" -msgstr "Fallo ao intentar borrar a carpeta do libro %(id)s, a ruta ten subcarpetas: %(path)s" +msgstr "Produciuse un erro ao eliminar o cartafol do libro %(id)s, o camiño ten subcartafoles: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" -msgstr "O borrado do libro %(id)s fallou: %(message)s" +msgstr "Produciuse un erro ao eliminar o libro %(id)s: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" -msgstr "Borrando o libro %(id)s, a ruta de libro non é válida: %(path)s" +msgstr "Eliminando o libro %(id)s só da base de datos, a ruta do libro na base de datos non é válida: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "O renomeado do título de: '%(src)s' a '%(dest)s' fallou co erro: %(error)s" +msgstr "O cambio de nome do autor de: '%(src)s' a '%(dest)s' fallou co erro: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" -msgstr "Ficheiro %(file)s non atopado en Google Drive" +msgstr "Non se atopou o ficheiro %(file)s en Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "O renomeado do título de: '%(src)s' a '%(dest)s' fallou co erro: %(error)s" +msgstr "O cambio de título de: '%(src)s' a '%(dest)s' produciu o erro: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" -msgstr "A ruta %(path)s do libro non se atopou en Google Drive" +msgstr "Non se atopou o camiño do libro %(path)s en Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" -msgstr "" +msgstr "Atopouse unha conta existente para este enderezo de correo electrónico" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" -msgstr "Este nome de usuario xa está en uso" +msgstr "Este nome de usuario xa está tomado" -#: cps/helper.py:702 -#, fuzzy +#: cps/helper.py:679 msgid "Invalid Email address format" -msgstr "Enderezo de correo non válido" +msgstr "Formato de enderezo de correo electrónico non válido" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" -msgstr "" +msgstr "O contrasinal non cumpre coas regras de validación do contrasinal" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" -msgstr "O módulo Python 'advocate' non está instalado pero se necesita para as cargas de cubertas" +msgstr "O módulo de Python 'advocate' non está instalado pero é necesario para cargas de portadas" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" -msgstr "Erro ao descargar a cuberta" +msgstr "Erro ao descargar a portada" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" -msgstr "Erro no formato da cuberta" +msgstr "Erro de formato de portada" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" -msgstr "Non ten permiso para acceder a localhost ou á rede local para as cargas de cubertas" +msgstr "Non tes permiso para acceder a localhost ou á rede local para cargas de portadas" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" -msgstr "Erro ao crear unha ruta para a cuberta" +msgstr "Produciuse un erro ao crear o camiño para a portada" + +#: cps/helper.py:889 +msgid "Cover-file is not a valid image file, or could not be stored" +msgstr "O ficheiro de portada non é un ficheiro de imaxe válido ou non se puido almacenar" + +#: cps/helper.py:900 +msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" +msgstr "Só se admiten ficheiros jpg/jpeg/png/webp/bmp como ficheiro de portada" #: cps/helper.py:912 -msgid "Cover-file is not a valid image file, or could not be stored" -msgstr "O arquivo de cuberta non é unha imaxe válida" - -#: cps/helper.py:923 -msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" -msgstr "Soamente se admiten como cuberta os arquivos jpg/jpeg/png/webp/bmp" - -#: cps/helper.py:935 msgid "Invalid cover file content" -msgstr "Contido do arquivo de cuberta non válido" +msgstr "Contido do ficheiro de portada non válido" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" -msgstr "Soamente se admiten como cuberta os arquivos jpg/jpeg" +msgstr "Só se admiten ficheiros jpg/jpeg como ficheiro de portada" -#: cps/helper.py:1011 cps/helper.py:1168 -#, fuzzy +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" -msgstr "Descubrir" +msgstr "Portada" + +#: cps/helper.py:1006 +msgid "UnRar binary file not found" +msgstr "Non se atopou o ficheiro binario UnRar" + +#: cps/helper.py:1017 +msgid "Error executing UnRar" +msgstr "Erro ao executar UnRar" + +#: cps/helper.py:1025 +msgid "Could not find the specified directory" +msgstr "Non se puido atopar o directorio especificado" #: cps/helper.py:1028 -msgid "UnRar binary file not found" -msgstr "Non se atopa o arquivo binario de UnRar" - -#: cps/helper.py:1039 -msgid "Error executing UnRar" -msgstr "Erro executando UnRar" - -#: cps/helper.py:1047 -msgid "Could not find the specified directory" -msgstr "" - -#: cps/helper.py:1050 msgid "Please specify a directory, not a file" -msgstr "" +msgstr "Especifique un directorio, non un ficheiro" -#: cps/helper.py:1064 -#, fuzzy +#: cps/helper.py:1042 msgid "Calibre binaries not viable" -msgstr "A base de datos non é modificable" +msgstr "Os binarios de calibre non son viables" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" -msgstr "" +msgstr "Faltan binarios de calibre: %(missing)s" -#: cps/helper.py:1075 -#, fuzzy, python-format +#: cps/helper.py:1053 +#, python-format msgid "Missing executable permissions: %(missing)s" -msgstr "Faltan permisos de execución" +msgstr "Faltan permisos executables: %(missing)s" -#: cps/helper.py:1080 -#, fuzzy +#: cps/helper.py:1058 msgid "Error executing Calibre" -msgstr "Erro executando UnRar" +msgstr "Erro ao executar Calibre" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" -msgstr "" +msgstr "Pon todos os libros en cola para a copia de seguridade dos metadatos" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" -msgstr "Por favor, accede a calibre-web dende unha ubicación que non sexa localhost para obter unha api_endpoint válida para o teu dispositivo Kobo" +msgstr "Accede a Calibre-Web desde un host non local para obter un api_endpoint válido para o dispositivo kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Configuración de Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" -msgstr "Rexistrado con %(provider)s" +msgstr "Rexístrate con %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" -msgstr "Iniciou sesión como : '%(nickname)s'" +msgstr "Éxito! Agora iniciaches sesión como: %(nickname)s" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" -msgstr "O enlace a %(oauth)s realizouse con éxito" +msgstr "Ligazón a %(oauth)s realizada correctamente" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" -msgstr "Acceso errado, ningún usuario enlazado coa conta OAuth" +msgstr "Fallou o inicio de sesión, ningún usuario vinculado coa conta OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" -msgstr "%(oauth)s desenlazado con éxito" +msgstr "Desvincular a %(oauth)s con éxito" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Erro ao desenlazar %(oauth)s" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" -msgstr "Non vinculado con %(oauth)s" +msgstr "Non vinculado a %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." -msgstr "Erro ao iniciar sesión con GitHub." +msgstr "Produciuse un erro ao iniciar sesión con GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." -msgstr "Erro ao obter a información do usuario de GitHub." +msgstr "Produciuse un erro ao obter a información do usuario de GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." -msgstr "Erro ao iniciar sesión con Google." +msgstr "Produciuse un erro ao iniciar sesión con Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." -msgstr "Erro ao obter información do usuario de Google." +msgstr "Produciuse un erro ao obter a información do usuario de Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." -msgstr "Erro en GitHub Oauth, por favor, volva a intentalo máis tarde." +msgstr "Erro de GitHub Oauth. Ténteo de novo máis tarde." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" -msgstr "Erro GitHub Oauth {}" +msgstr "Erro de GitHub Oauth: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." -msgstr "Erro en Google Oauth, por favor volva a intentalo máis tarde." +msgstr "Erro de Google Oauth. Téntao de novo máis tarde." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" -msgstr "Erro Google Oauth {}" +msgstr "Erro de Google Oauth: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" -msgstr "{} Estrelas" +msgstr "{} estrelas" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" -msgstr "Inicio de sesión" +msgstr "Iniciar sesión" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" -msgstr "Token non atopado" +msgstr "Non se atopou o token" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" -msgstr "O token expirou" +msgstr "O token caducou" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" -msgstr "Correcto! Por favor volte ao seu dispositivo" +msgstr "Éxito! Volve ao teu dispositivo" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Libros" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Mostrar libros recentes" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Libros populares" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Mostrar libros populares" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Libros descargados" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Mostrar libros descargados" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Libros mellor valorados" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Mostrar libros mellor valorados" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Libros lidos" -#: cps/render_template.py:64 -#, fuzzy +#: cps/render_template.py:63 msgid "Show Read and Unread" -msgstr "Mostrar lidos e non lidos" +msgstr "Mostrar lido e non lido" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" -msgstr "Libros non lidos" +msgstr "Libros sen ler" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Mostrar non lidos" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Descubrir" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" -msgstr "Mostrar libros ao chou" +msgstr "Mostrar libros aleatorios" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Categorías" -#: cps/render_template.py:74 cps/templates/user_table.html:158 -#, fuzzy +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" -msgstr "Mostrar selección de categorías" +msgstr "Mostrar sección de categorías" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" -msgstr "Series" +msgstr "Serie" -#: cps/render_template.py:77 cps/templates/user_table.html:157 -#, fuzzy +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" -msgstr "Mostrar selección de series" +msgstr "Mostrar Sección Serie" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autores" -#: cps/render_template.py:80 cps/templates/user_table.html:160 -#, fuzzy +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" -msgstr "Mostrar selección de autores" +msgstr "Mostrar a sección Autor" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Editores" -#: cps/render_template.py:84 cps/templates/user_table.html:163 -#, fuzzy +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" -msgstr "Mostrar selección de editores" +msgstr "Mostrar a sección de editores" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Linguas" -#: cps/render_template.py:88 cps/templates/user_table.html:155 -#, fuzzy +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" -msgstr "Mostrar selección de linguas" +msgstr "Mostrar a sección de idioma" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Valoracións" -#: cps/render_template.py:91 cps/templates/user_table.html:164 -#, fuzzy +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" -msgstr "Mostrar selección de valoracións" +msgstr "Mostrar a sección de valoracións" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" -msgstr "Formatos de arquivo" +msgstr "Formatos de ficheiro" -#: cps/render_template.py:94 cps/templates/user_table.html:165 -#, fuzzy +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" -msgstr "Mostrar selección de formatos de arquivo" +msgstr "Mostrar a sección de formatos de ficheiro" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" -msgstr "Libros arquivados" +msgstr "Libros Arquivados" -#: cps/render_template.py:98 cps/templates/user_table.html:166 -#, fuzzy +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "Mostrar libros arquivados" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Lista de libros" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" -msgstr "Mostrar lista de libros" +msgstr "Mostrar a lista de libros" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Buscar" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " -msgstr "Publicado despóis de " +msgstr "Publicado despois " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " -msgstr "Publicado antes de " +msgstr "Publicado antes " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Valoración <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Valoración >= %(rating)s" -#: cps/search.py:221 -#, fuzzy, python-format +#: cps/search.py:234 +#, python-format msgid "Read Status = '%(status)s'" msgstr "Estado de lectura = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" -msgstr "Erro na busca de columnas personalizadas, por favor reinicia Calibre-Web" +msgstr "Produciuse un erro ao buscar columnas personalizadas, reinicie Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" -msgstr "Búsqueda avanzada" +msgstr "Busca avanzada" #: cps/shelf.py:49 cps/shelf.py:111 msgid "Invalid shelf specified" -msgstr "Andel especificado non válido" +msgstr "Especificouse unha estantería non válida" #: cps/shelf.py:55 msgid "Sorry you are not allowed to add a book to that shelf" -msgstr "Sentímolo, non ten permisos para engdir un libro ao andel: %(shelfname)s" +msgstr "Sentímolo, non tes permiso para engadir un libro a ese estante" #: cps/shelf.py:64 #, python-format msgid "Book is already part of the shelf: %(shelfname)s" -msgstr "O libro xa forma parte do andel: %(shelfname)s" +msgstr "O libro xa forma parte do estante: %(shelfname)s" #: cps/shelf.py:77 #, python-format msgid "%(book_id)s is a invalid Book Id. Could not be added to Shelf" -msgstr "" +msgstr "%(book_id)s é un ID de libro non válido. Non se puido engadir ao estante" #: cps/shelf.py:97 #, python-format msgid "Book has been added to shelf: %(sname)s" -msgstr "O libro engadiuse ao andel: %(sname)s" +msgstr "Engadiuse o libro ao estante: %(sname)s" #: cps/shelf.py:116 msgid "You are not allowed to add a book to the shelf" -msgstr "Non ten permiso para engadir un libro ao andel" +msgstr "Non tes permiso para engadir un libro ao estante" #: cps/shelf.py:134 #, python-format msgid "Books are already part of the shelf: %(name)s" -msgstr "Os libros xa forman parte do andel: %(name)s" +msgstr "Os libros xa forman parte do estante: %(name)s" #: cps/shelf.py:146 #, python-format msgid "Books have been added to shelf: %(sname)s" -msgstr "Os libros engadíronse ao andel: %(sname)s" +msgstr "Engadíronse libros ao estante: %(sname)s" #: cps/shelf.py:153 #, python-format msgid "Could not add books to shelf: %(sname)s" -msgstr "Non se pudideron engadir libros ao andel: %(sname)s" +msgstr "Non se puideron engadir libros ao estante: %(sname)s" #: cps/shelf.py:199 #, python-format msgid "Book has been removed from shelf: %(sname)s" -msgstr "O libro eliminouse do andel: %(sname)s" +msgstr "Eliminouse o libro do estante: %(sname)s" #: cps/shelf.py:208 msgid "Sorry you are not allowed to remove a book from this shelf" -msgstr "Síntoo, non ten permiso para quitar un libro do andel" +msgstr "Sentímolo, non tes permiso para eliminar un libro deste estante" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" -msgstr "Crear un andel" +msgstr "Crear un estante" #: cps/shelf.py:226 msgid "Sorry you are not allowed to edit this shelf" -msgstr "Síntoo, non ten permiso para editar o andel: %(sname)s" +msgstr "Sentímolo, non tes permiso para editar este estante" #: cps/shelf.py:228 msgid "Edit a shelf" -msgstr "Editar un andel" +msgstr "Editar un estante" #: cps/shelf.py:237 msgid "Error deleting Shelf" -msgstr "Erro borrando o estante" +msgstr "Produciuse un erro ao eliminar o estante" #: cps/shelf.py:239 msgid "Shelf successfully deleted" -msgstr "Andel eliminado con éxito" +msgstr "Eliminouse correctamente o estante" #: cps/shelf.py:289 #, python-format msgid "Change order of Shelf: '%(name)s'" -msgstr "Cambiar a orde do andel: '%(name)s'" +msgstr "Cambiar a orde do estante: '%(name)s'" #: cps/shelf.py:324 msgid "Sorry you are not allowed to create a public shelf" -msgstr "Síntoo, non ten permiso para crear un andel público" +msgstr "Sentímolo, non tes permiso para crear un estante público" #: cps/shelf.py:341 #, python-format msgid "Shelf %(title)s created" -msgstr "Andel %(title)s creado" +msgstr "Creouse o estante %(title)s" #: cps/shelf.py:344 #, python-format msgid "Shelf %(title)s changed" -msgstr "Andel %(title)s cambiado" +msgstr "O estante %(title)s cambiou" #: cps/shelf.py:358 msgid "There was an error" -msgstr "Ocorreu un erro" +msgstr "Houbo un erro" #: cps/shelf.py:380 #, python-format msgid "A public shelf with the name '%(title)s' already exists." -msgstr "Xa existe un andel público co nome '%(title)s'." +msgstr "Xa existe un estante público co nome '%(title)s'." #: cps/shelf.py:391 #, python-format msgid "A private shelf with the name '%(title)s' already exists." -msgstr "Xa existe un andel privado co nome '%(title)s'." +msgstr "Xa existe un estante privado co nome '%(title)s'." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" -msgstr "Andel: '%(name)s'" +msgstr "Estante: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" -msgstr "Erro ao abrir un andel. O andel non existe ou non se pode acceder" +msgstr "Produciuse un erro ao abrir o estante. O estante non existe ou non é accesible" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tarefas" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" -msgstr "Esperando" +msgstr "Agardando" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Fallou" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" -msgstr "Comezado" +msgstr "Comezou" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" -msgstr "Rematado" +msgstr "Rematou" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" -msgstr "Rematado" +msgstr "Rematou" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Cancelado" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Estado descoñecido" #: cps/updater.py:433 cps/updater.py:444 cps/updater.py:545 cps/updater.py:560 msgid "Unexpected data while reading update information" -msgstr "Dato inesperado mentres líase a información de actualización" +msgstr "Datos inesperados ao ler a información de actualización" #: cps/updater.py:440 cps/updater.py:552 msgid "No update available. You already have the latest version installed" -msgstr "Actualización non dispoñiible. Xa tes instalada a versión máis recente" +msgstr "Non hai ningunha actualización dispoñible. Xa tes instalada a última versión" #: cps/updater.py:458 msgid "A new update is available. Click on the button below to update to the latest version." -msgstr "Unha nova actualización está dispoñible. Preme no botón inferior para actualizar á versión máis reciente." +msgstr "Hai unha nova actualización dispoñible. Fai clic no botón de abaixo para actualizar á última versión." #: cps/updater.py:476 msgid "Could not fetch update information" -msgstr "Non se pode conseguir información sobre a actualización" +msgstr "Non se puido recuperar a información de actualización" #: cps/updater.py:486 msgid "Click on the button below to update to the latest stable version." -msgstr "Preme no botón de abaixo para actualizar á última versión estable." +msgstr "Fai clic no botón de abaixo para actualizar á última versión estable." #: cps/updater.py:495 cps/updater.py:509 cps/updater.py:520 #, python-format msgid "A new update is available. Click on the button below to update to version: %(version)s" -msgstr "Hai unha nova actualización dispoñible. Preme no botón de abaixo para actualizar á versión: %(version)s" +msgstr "Hai unha nova actualización dispoñible. Fai clic no botón de abaixo para actualizar á versión: %(version)s" #: cps/updater.py:538 msgid "No release information available" -msgstr "Non hai información do lanzamento dispoñible" +msgstr "Non hai información de lanzamento dispoñible" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" -msgstr "Descubrir (Libros ao chou)" +msgstr "Descubre (Libros aleatorios)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" -msgstr "Libros populares (os máis descargados)" +msgstr "Libros populares (máis descargados)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Libros descargados por %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" -msgstr "Autor/es: %(name)s" +msgstr "Autor: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" -msgstr "Editor/es: %(name)s" +msgstr "Editor: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" -msgstr "Series: %(serie)s" +msgstr "Serie: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Valoración: Ningunha" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Valoración: %(rating)s estrelas" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" -msgstr "Formato do arquivo: %(format)s" +msgstr "Formato de ficheiro: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Categoría: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" -msgstr "Lingua: %(name)s" +msgstr "Idioma: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Descargas" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Lista de valoracións" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" -msgstr "Lista de formatos" - -#: cps/web.py:1249 -#, fuzzy -msgid "Please configure the SMTP mail settings first..." -msgstr "Configura primeiro os parámetros do servidor SMTP..." - -#: cps/web.py:1256 -#, python-format -msgid "Success! Book queued for sending to %(eReadermail)s" -msgstr "Libro posto na cola de envío a %(eReadermail)s" +msgstr "Lista de formatos de ficheiro" #: cps/web.py:1259 +msgid "Please configure the SMTP mail settings first..." +msgstr "Configure primeiro a configuración do correo SMTP..." + +#: cps/web.py:1265 +#, python-format +msgid "Success! Book queued for sending to %(eReadermail)s" +msgstr "Éxito! Libro en cola para enviar a %(eReadermail)s" + +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" -msgstr "Oh, oh! Houbo un erro no envío do libro: %(res)s" +msgstr "Vaia! Produciuse un erro ao enviar o libro: %(res)s" -#: cps/web.py:1261 -#, fuzzy +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." -msgstr "Por favor actualiza o teu perfil co enderezo de correo do teu kindle..." +msgstr "Vaia! Actualiza o teu perfil cun correo electrónico de eReader válido." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" -msgstr "" +msgstr "Agarde un minuto para rexistrar o seguinte usuario" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" -msgstr "Rexistro" +msgstr "Rexístrate" -#: cps/web.py:1281 cps/web.py:1385 -#, fuzzy +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" -msgstr "O servidor de correo non está configurado, por favor, avisa ao teu administrador!" +msgstr "Produciuse un erro de conexión ao back-end do limitador. Ponte en contacto co teu administrador" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." -msgstr "O servidor de correo non está configurado, por favor, avisa ao teu administrador!" +msgstr "Vaia! O servidor de correo electrónico non está configurado, póñase en contacto co seu administrador." -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." -msgstr "O seu correo electrónico non está permitido para rexistrarse" +msgstr "Vaia! O teu correo electrónico non está permitido." -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." -msgstr "Mandouse un correo electrónico de verificación á súa conta de correo." +msgstr "Éxito! Enviouse o correo electrónico de confirmación." -#: cps/web.py:1368 cps/web.py:1391 -#, fuzzy +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "Non se pode activar a autenticación LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" -msgstr "" +msgstr "Agarde un minuto antes do próximo inicio de sesión" -#: cps/web.py:1400 -#, fuzzy, python-format +#: cps/web.py:1408 +#, python-format msgid "you are now logged in as: '%(nickname)s'" -msgstr "Iniciou sesión como : '%(nickname)s'" +msgstr "agora iniciaches sesión como: '%(nickname)s'" -#: cps/web.py:1407 -#, fuzzy, python-format +#: cps/web.py:1415 +#, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" -msgstr "Fallback login como: '%(nickname)s', non se pode acceder ao servidor LDAP ou usuario descoñecido" +msgstr "Inicio de sesión alternativo como: '%(nickname)s', servidor LDAP non accesible ou usuario descoñecido" -#: cps/web.py:1412 -#, fuzzy, python-format +#: cps/web.py:1420 +#, python-format msgid "Could not login: %(message)s" -msgstr "Non se puido entrar: %(message)s" +msgstr "Non se puido iniciar sesión: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 -#, fuzzy +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" -msgstr "Usuario ou contrasinal no válido" +msgstr "Nome de usuario ou contrasinal incorrecto" -#: cps/web.py:1423 -#, fuzzy +#: cps/web.py:1431 msgid "New Password was sent to your email address" -msgstr "Unha nova contrasinal enviouse ao seu enderezo de correo electrónico" +msgstr "Enviouse un novo contrasinal ao teu enderezo de correo electrónico" -#: cps/web.py:1427 -#, fuzzy +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." -msgstr "Sucedeu un erro descoñecido. Por favor volva a intentalo máis tarde." - -#: cps/web.py:1429 -#, fuzzy -msgid "Please enter valid username to reset password" -msgstr "Por favor, introduce un usuario válido para restablecer a contrasinal" +msgstr "Produciuse un erro descoñecido. Téntao de novo máis tarde." #: cps/web.py:1437 -#, fuzzy, python-format -msgid "You are now logged in as: '%(nickname)s'" -msgstr "Iniciou sesión como : '%(nickname)s'" +msgid "Please enter valid username to reset password" +msgstr "Introduza un nome de usuario válido para restablecer o contrasinal" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1445 +#, python-format +msgid "You are now logged in as: '%(nickname)s'" +msgstr "Agora iniciaches sesión como: '%(nickname)s'" + +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Perfil de %(name)s" -#: cps/web.py:1511 -#, fuzzy +#: cps/web.py:1526 msgid "Success! Profile Updated" -msgstr "Perfil actualizado" +msgstr "Éxito! Perfil actualizado" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." -msgstr "Atopada unha conta existente para ese enderezo de correo electrónico" +msgstr "Vaia! Xa existe unha conta para este correo electrónico." #: cps/services/gmail.py:59 msgid "Found no valid gmail.json file with OAuth information" -msgstr "Non se atopou ningún arquivo gmail.json válido con información OAuth" +msgstr "Non se atopou ningún ficheiro gmail.json válido con información de OAuth" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "Eliminar o contido do cartafol temporal" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" -msgstr "Enviar ao Kindle %(book)s" +msgstr "%(book)s envía a E-Reader" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Non se atopou Calibre ebook-convert %(tool)s" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" -msgstr "O formato %(format)s non se atopou no disco" +msgstr "Non se atopou o formato %(format)s no disco" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" -msgstr "O conversor de Ebook fallou cun erro descoñecido" +msgstr "Fallou o conversor de libros electrónicos cun erro descoñecido" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-converter fallou: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" -msgstr "Arquivo convertido non atopado, ou máis dun arquivo no directorio %(folder)s" +msgstr "Non se atopou o ficheiro convertido ou hai máis dun ficheiro no cartafol %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Fallou Ebook-converter: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" -msgstr "Calibre fallou co erro: %(error)s" +msgstr "Fallou o calibre co erro: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Fallou o convertidor de libros electrónicos: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" -msgstr "Convertir" +msgstr "Converter" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" -msgstr "Reconectando a base de datos de Calibre" +msgstr "Reconectando a base de datos Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "Correo electrónico" #: cps/tasks/metadata_backup.py:34 -#, fuzzy msgid "Backing up Metadata" -msgstr "editar metadatos" - -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" +msgstr "Facendo copia de seguranza dos metadatos" #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" -msgstr "Xeradas %(count)s miniaturas de cubertas" +msgstr "Xeráronse %(count)s Miniaturas de portada" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" -msgstr "Miniaturas de cubertas" +msgstr "Miniaturas de portada" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" -msgstr "Xeradas {0} miniaturas de series" +msgstr "Xeráronse {0} miniaturas de series" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" -msgstr "Limpando caché de miniaturas de cubertas" +msgstr "Borrando a caché de miniaturas da portada" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" -msgstr "Subir arquivo" +msgstr "Cargar" #: cps/templates/admin.html:9 msgid "Users" @@ -1582,24 +1557,24 @@ msgstr "Nome de usuario" #: cps/templates/admin.html:14 cps/templates/register.html:14 #: cps/templates/user_edit.html:15 cps/templates/user_table.html:135 msgid "Email" -msgstr "Enderezo de correo electrónico" +msgstr "Correo electrónico" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" -msgstr "Enviar ao enderezo de correo electrónico do Kindle" +msgstr "Enviar ao correo electrónico eReader" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" -msgstr "Administrador" +msgstr "Admin" #: cps/templates/admin.html:18 cps/templates/login.html:13 #: cps/templates/login.html:14 cps/templates/user_edit.html:23 msgid "Password" msgstr "Contrasinal" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Descargar" @@ -1618,11 +1593,11 @@ msgstr "Editar" #: cps/templates/modal_dialogs.html:116 cps/templates/user_edit.html:67 #: cps/templates/user_table.html:149 msgid "Delete" -msgstr "Borrar" +msgstr "Eliminar" #: cps/templates/admin.html:26 msgid "Public Shelf" -msgstr "Andel público" +msgstr "Estante Público" #: cps/templates/admin.html:55 msgid "Import LDAP Users" @@ -1630,11 +1605,11 @@ msgstr "Importar usuarios LDAP" #: cps/templates/admin.html:62 msgid "Email Server Settings" -msgstr "Axustes do servidor de correo electrónico" +msgstr "Configuración do servidor de correo electrónico" #: cps/templates/admin.html:67 cps/templates/email_edit.html:31 msgid "SMTP Hostname" -msgstr "Servidor SMTP" +msgstr "Nome de host SMTP" #: cps/templates/admin.html:71 cps/templates/email_edit.html:35 msgid "SMTP Port" @@ -1642,25 +1617,24 @@ msgstr "Porto SMTP" #: cps/templates/admin.html:75 cps/templates/email_edit.html:39 msgid "Encryption" -msgstr "Cifraxe" +msgstr "Cifrado" #: cps/templates/admin.html:79 cps/templates/email_edit.html:47 msgid "SMTP Login" -msgstr "Login SMTP" +msgstr "Inicio de sesión SMTP" #: cps/templates/admin.html:83 cps/templates/admin.html:94 #: cps/templates/email_edit.html:55 msgid "From Email" -msgstr "Dende o correo electrónico" +msgstr "Desde o correo electrónico" #: cps/templates/admin.html:90 -#, fuzzy msgid "Email Service" msgstr "Servizo de correo electrónico" #: cps/templates/admin.html:91 msgid "Gmail via Oauth2" -msgstr "Gmail por Oauth2" +msgstr "Gmail a través de Oauth2" #: cps/templates/admin.html:106 msgid "Configuration" @@ -1680,7 +1654,7 @@ msgstr "Porto" #: cps/templates/admin.html:122 msgid "External Port" -msgstr "Porto externo" +msgstr "Porto Externo" #: cps/templates/admin.html:129 cps/templates/config_view_edit.html:28 msgid "Books per Page" @@ -1696,19 +1670,19 @@ msgstr "Navegación anónima" #: cps/templates/admin.html:141 msgid "Public Registration" -msgstr "Rexistro público" +msgstr "Rexistro Público" #: cps/templates/admin.html:145 msgid "Magic Link Remote Login" -msgstr "Acceso remoto mediante enlace máxico" +msgstr "Acceso remoto Magic Link" #: cps/templates/admin.html:149 msgid "Reverse Proxy Login" -msgstr "Acceso mediante Proxy inverso" +msgstr "Inicio de sesión de proxy inverso" #: cps/templates/admin.html:154 cps/templates/config_edit.html:172 msgid "Reverse Proxy Header Name" -msgstr "Nome de cabeceira do Proxy inverso" +msgstr "Nome da cabeceira do proxy inverso" #: cps/templates/admin.html:159 msgid "Edit Calibre Database Configuration" @@ -1716,11 +1690,11 @@ msgstr "Editar a configuración da base de datos de Calibre" #: cps/templates/admin.html:160 msgid "Edit Basic Configuration" -msgstr "Editar a configuración básica" +msgstr "Editar configuración básica" #: cps/templates/admin.html:161 msgid "Edit UI Configuration" -msgstr "Editar a configuración da interfaz de usuario" +msgstr "Editar a configuración da interface de usuario" #: cps/templates/admin.html:167 msgid "Scheduled Tasks" @@ -1729,7 +1703,7 @@ msgstr "Tarefas programadas" #: cps/templates/admin.html:170 cps/templates/schedule_edit.html:12 #: cps/templates/tasks.html:18 msgid "Start Time" -msgstr "Hora de comezo das tarefas" +msgstr "Hora de inicio" #: cps/templates/admin.html:174 cps/templates/schedule_edit.html:20 msgid "Maximum Duration" @@ -1737,24 +1711,24 @@ msgstr "Duración máxima das tarefas" #: cps/templates/admin.html:178 cps/templates/schedule_edit.html:29 msgid "Generate Thumbnails" -msgstr "Xerar as miniaturas das cubertas dos libros" +msgstr "Xerar miniaturas" #: cps/templates/admin.html:182 msgid "Generate series cover thumbnails" -msgstr "Xerar as miniaturas das cubertas das series" +msgstr "Xera miniaturas de portadas de serie" #: cps/templates/admin.html:186 cps/templates/admin.html:208 #: cps/templates/schedule_edit.html:37 msgid "Reconnect Calibre Database" -msgstr "Reconectar á librería de Calibre" +msgstr "Reconectar a base de datos Calibre" #: cps/templates/admin.html:190 cps/templates/schedule_edit.html:41 msgid "Generate Metadata Backup Files" -msgstr "" +msgstr "Xerar ficheiros de copia de seguridade de metadatos" #: cps/templates/admin.html:197 msgid "Refresh Thumbnail Cache" -msgstr "Refrescar a caché de miniaturas das cubertas" +msgstr "Actualizar caché de miniaturas" #: cps/templates/admin.html:203 msgid "Administration" @@ -1762,11 +1736,11 @@ msgstr "Administración" #: cps/templates/admin.html:204 msgid "Download Debug Package" -msgstr "Descargar o paquete de depuración" +msgstr "Descargar paquete de depuración" #: cps/templates/admin.html:205 msgid "View Logs" -msgstr "Ver arquivos de rexistro" +msgstr "Ver rexistros" #: cps/templates/admin.html:211 msgid "Restart" @@ -1774,11 +1748,11 @@ msgstr "Reiniciar" #: cps/templates/admin.html:212 msgid "Shutdown" -msgstr "Apagar" +msgstr "Apagado" #: cps/templates/admin.html:221 msgid "Version Information" -msgstr "Información de versión" +msgstr "Información da versión" #: cps/templates/admin.html:225 msgid "Version" @@ -1794,7 +1768,7 @@ msgstr "Versión actual" #: cps/templates/admin.html:239 msgid "Check for Update" -msgstr "Comprobar actualizacións" +msgstr "Comprobar a actualización" #: cps/templates/admin.html:240 msgid "Perform Update" @@ -1802,7 +1776,7 @@ msgstr "Realizar actualización" #: cps/templates/admin.html:253 msgid "Are you sure you want to restart?" -msgstr "De verdade queres reiniciar?" +msgstr "Estás seguro de que queres reiniciar?" #: cps/templates/admin.html:258 cps/templates/admin.html:272 #: cps/templates/admin.html:292 cps/templates/config_db.html:82 @@ -1810,193 +1784,261 @@ msgid "OK" msgstr "Vale" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Cancelar" #: cps/templates/admin.html:271 msgid "Are you sure you want to shutdown?" -msgstr "De verdade quere deter?" +msgstr "Estás seguro de que queres apagar?" #: cps/templates/admin.html:283 msgid "Updating, please do not reload this page" -msgstr "Actualizando. Por favor, non recargue a páxina" +msgstr "Actualizando, non volvas cargar esta páxina" #: cps/templates/author.html:15 msgid "via" -msgstr "via" +msgstr "vía" #: cps/templates/author.html:23 msgid "In Library" -msgstr "Na Librería" +msgstr "En Biblioteca" #: cps/templates/author.html:26 cps/templates/index.html:74 #: cps/templates/search.html:31 cps/templates/shelf.html:20 msgid "Sort according to book date, newest first" -msgstr "Ordear pola data do libro, máis recente primeiro" +msgstr "Ordena segundo a data do libro, primeiro o máis novo" #: cps/templates/author.html:27 cps/templates/index.html:75 #: cps/templates/search.html:32 cps/templates/shelf.html:21 msgid "Sort according to book date, oldest first" -msgstr "Ordear pola data do libro, máis antigo primeiro" +msgstr "Ordena segundo a data do libro, primeiro o máis antigo" #: cps/templates/author.html:28 cps/templates/index.html:76 #: cps/templates/search.html:33 cps/templates/shelf.html:22 msgid "Sort title in alphabetical order" -msgstr "Ordear polo título en orden alfabético" +msgstr "Ordena o título por orde alfabética" #: cps/templates/author.html:29 cps/templates/index.html:77 #: cps/templates/search.html:34 cps/templates/shelf.html:23 msgid "Sort title in reverse alphabetical order" -msgstr "Ordear polo título en orde alfabético invertido" +msgstr "Ordena o título en orde alfabética inversa" #: cps/templates/author.html:30 cps/templates/index.html:80 #: cps/templates/search.html:37 cps/templates/shelf.html:26 msgid "Sort according to publishing date, newest first" -msgstr "Ordear pola data de publicación, máis recente primeiro" +msgstr "Ordena segundo a data de publicación, primeiro o máis novo" #: cps/templates/author.html:31 cps/templates/index.html:81 #: cps/templates/search.html:38 cps/templates/shelf.html:27 msgid "Sort according to publishing date, oldest first" -msgstr "Ordear pola data de publicación, máis antigo primeiro" +msgstr "Ordena segundo a data de publicación, primeiro o máis antigo" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "reducir" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" -msgstr "Máis de" +msgstr "Máis por" + +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Libro %(index)s de %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Linguaxe" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Editora" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publicado" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Descrición:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Anterior" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "A continuación" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Non se atoparon resultados" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Inicio" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Busca na biblioteca" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Pechar sesión" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" #: cps/templates/book_edit.html:11 msgid "Delete Book" -msgstr "Borrar libro" +msgstr "Eliminar libro" #: cps/templates/book_edit.html:14 msgid "Delete formats:" -msgstr "Borrar formatos:" +msgstr "Eliminar formatos:" #: cps/templates/book_edit.html:25 msgid "Convert book format:" -msgstr "Convertir formato de libro:" +msgstr "Converter formato de libro:" #: cps/templates/book_edit.html:30 msgid "Convert from:" -msgstr "Convertir dende:" +msgstr "Converter de:" #: cps/templates/book_edit.html:32 cps/templates/book_edit.html:39 msgid "select an option" -msgstr "seleccionar unha opción" +msgstr "seleccione unha opción" #: cps/templates/book_edit.html:37 msgid "Convert to:" -msgstr "Convertir a:" +msgstr "Converter a:" #: cps/templates/book_edit.html:46 msgid "Convert book" -msgstr "Convertir libro" +msgstr "Converter libro" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Cargando..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Pechar" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Erro" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Carga feita, procesando, espera..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Formato de carga" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Título do libro" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 -msgid "Description" -msgstr "Descrición" - -#: cps/templates/book_edit.html:73 -msgid "Identifiers" -msgstr "Identificadores" - -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 -msgid "Identifier Type" -msgstr "Tipo de identificador" - -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 -msgid "Identifier Value" -msgstr "Valor de identificador" - -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 -#: cps/templates/user_table.html:24 -msgid "Remove" -msgstr "Borrar" - -#: cps/templates/book_edit.html:83 -msgid "Add Identifier" -msgstr "Engadir identificador" - -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 msgid "Tags" msgstr "Etiquetas" -#: cps/templates/book_edit.html:95 +#: cps/templates/book_edit.html:90 msgid "Series ID" msgstr "ID de serie" -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Valoración" - -#: cps/templates/book_edit.html:104 -msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" -msgstr "Obter portada de URL (JPEG, a portada descargarase e almacenarase na base de datos)" - -#: cps/templates/book_edit.html:108 -msgid "Upload Cover from Local Disk" -msgstr "Cargar portada dende un disco local" - -#: cps/templates/book_edit.html:113 +#: cps/templates/book_edit.html:93 msgid "Published Date" msgstr "Data de publicación" -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Editor" +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Valoración" -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Lingua" +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 +msgid "Description" +msgstr "Descrición" -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:118 +msgid "Identifiers" +msgstr "Identificadores" + +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 +msgid "Identifier Type" +msgstr "Tipo de identificador" + +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 +msgid "Identifier Value" +msgstr "Valor do identificador" + +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 +#: cps/templates/user_table.html:24 +msgid "Remove" +msgstr "Eliminar" + +#: cps/templates/book_edit.html:129 +msgid "Add Identifier" +msgstr "Engadir identificador" + +#: cps/templates/book_edit.html:133 +msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" +msgstr "Obter a portada do URL (JPEG: a imaxe descargarase e almacenarase na base de datos)" + +#: cps/templates/book_edit.html:137 +msgid "Upload Cover from Local Disk" +msgstr "Cargar portada desde o disco local" + +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Sí" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Non" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Cargar formato" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" -msgstr "Ver libro ao gardar" +msgstr "Ver Libro en Gardar" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Obter metadatos" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2004,69 +2046,63 @@ msgstr "Obter metadatos" msgid "Save" msgstr "Gardar" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" -msgstr "Palabra chave" - -#: cps/templates/book_edit.html:234 -msgid "Search keyword" -msgstr "Buscar por palabras chave " +msgstr "Palabra clave" #: cps/templates/book_edit.html:240 -msgid "Click the cover to load metadata to the form" -msgstr "Premer na portada para cargar los metadatos no formulario" +msgid "Search keyword" +msgstr "Buscar palabra clave" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:246 +msgid "Click the cover to load metadata to the form" +msgstr "Fai clic na portada para cargar metadatos no formulario" + +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Cargando..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Pechar" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" -msgstr "Orixe" +msgstr "Fonte" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" -msgstr "Erro na busca!" +msgstr "Erro de busca!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." -msgstr "Non se atoparon resultados! Por favor intenta con outra palabra chave." +msgstr "Non se atopou ningún resultado! Proba con outra palabra clave." #: cps/templates/book_table.html:12 cps/templates/book_table.html:69 #: cps/templates/user_table.html:14 cps/templates/user_table.html:77 #: cps/templates/user_table.html:100 msgid "This Field is Required" -msgstr "Este campo é necesario" +msgstr "Este campo é obrigatorio" #: cps/templates/book_table.html:37 msgid "Merge selected books" -msgstr "Unir libros seleccionados" +msgstr "Combina os libros seleccionados" #: cps/templates/book_table.html:38 cps/templates/user_table.html:124 msgid "Remove Selections" -msgstr "Eliminar selección" +msgstr "Eliminar seleccións" #: cps/templates/book_table.html:41 msgid "Exchange author and title" -msgstr "Intercambiar autor e título" +msgstr "Intercambio de autor e título" #: cps/templates/book_table.html:47 msgid "Update Title Sort automatically" -msgstr "Actualizar orde do título automáticamente" +msgstr "Actualizar o Título para Ordenar automaticamente" #: cps/templates/book_table.html:51 msgid "Update Author Sort automatically" -msgstr "Actualizar orde do autor automáticamente" +msgstr "Actualizar o Autor para Ordenar automaticamente" #: cps/templates/book_table.html:63 cps/templates/book_table.html:69 msgid "Enter Title" -msgstr "Introduce título" +msgstr "Introduza o título" #: cps/templates/book_table.html:63 cps/templates/config_view_edit.html:24 #: cps/templates/shelf_edit.html:8 @@ -2075,39 +2111,39 @@ msgstr "Título" #: cps/templates/book_table.html:64 msgid "Enter Title Sort" -msgstr "Introduce a orde do título" +msgstr "Introduza o Título para Ordenar" #: cps/templates/book_table.html:64 msgid "Title Sort" -msgstr "Orde do título" +msgstr "Título para Ordenar" #: cps/templates/book_table.html:65 msgid "Enter Author Sort" -msgstr "Introduce orde do autor" +msgstr "Introduce o Autor para Ordenar" #: cps/templates/book_table.html:65 msgid "Author Sort" -msgstr "Orde do autor" +msgstr "Autor para Ordenar" #: cps/templates/book_table.html:66 msgid "Enter Authors" -msgstr "Introduce os autores" +msgstr "Introducir Autores" #: cps/templates/book_table.html:67 msgid "Enter Categories" -msgstr "Introduce as categorías" +msgstr "Introduza categorías" #: cps/templates/book_table.html:68 msgid "Enter Series" -msgstr "Introduce as series" +msgstr "Introduza as series" #: cps/templates/book_table.html:69 msgid "Series Index" -msgstr "Índice da serie" +msgstr "Índice de series" #: cps/templates/book_table.html:70 msgid "Enter Languages" -msgstr "Introduce as linguas" +msgstr "Introduza Idiomas" #: cps/templates/book_table.html:71 msgid "Publishing Date" @@ -2115,11 +2151,11 @@ msgstr "Data de publicación" #: cps/templates/book_table.html:72 msgid "Enter Publishers" -msgstr "Introduce os Editores" +msgstr "Entra Editores" #: cps/templates/book_table.html:73 msgid "Enter comments" -msgstr "Introducir os comentarios" +msgstr "Introduce comentarios" #: cps/templates/book_table.html:73 msgid "Comments" @@ -2131,31 +2167,31 @@ msgstr "Estado do arquivo" #: cps/templates/book_table.html:77 cps/templates/search_form.html:42 msgid "Read Status" -msgstr "Leer estado" +msgstr "Estado de lectura" #: cps/templates/book_table.html:80 cps/templates/book_table.html:82 #: cps/templates/book_table.html:84 cps/templates/book_table.html:86 #: cps/templates/book_table.html:90 cps/templates/book_table.html:92 #: cps/templates/book_table.html:96 msgid "Enter " -msgstr "Introduce " +msgstr "Entra " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" -msgstr "De verdade estás seguro?" +msgstr "Estás realmente seguro?" #: cps/templates/book_table.html:117 msgid "Books with Title will be merged from:" -msgstr "Libros co título uniranse de:" +msgstr "Os libros con título combinaranse de:" #: cps/templates/book_table.html:121 msgid "Into Book with Title:" -msgstr "No libro co título:" +msgstr "No libro con título:" #: cps/templates/book_table.html:126 msgid "Merge" -msgstr "Unir" +msgstr "Combinar" #: cps/templates/config_db.html:12 msgid "Location of Calibre Database" @@ -2163,11 +2199,11 @@ msgstr "Localización da base de datos Calibre" #: cps/templates/config_db.html:21 msgid "Separate Book Files from Library" -msgstr "" +msgstr "Separa os ficheiros de libros da biblioteca" #: cps/templates/config_db.html:34 msgid "Use Google Drive?" -msgstr "Usar Google Drive?" +msgstr "Usa Google Drive?" #: cps/templates/config_db.html:39 msgid "Authenticate Google Drive" @@ -2175,11 +2211,11 @@ msgstr "Autenticar Google Drive" #: cps/templates/config_db.html:44 msgid "Google Drive Calibre folder" -msgstr "Carpeta de Google Drive para Calibre" +msgstr "Cartafol Google Drive Calibre" #: cps/templates/config_db.html:52 msgid "Metadata Watch Channel ID" -msgstr "Metadata Watch Channel ID" +msgstr "ID da canle de visualización de metadatos" #: cps/templates/config_db.html:55 msgid "Revoke" @@ -2187,7 +2223,7 @@ msgstr "Revogar" #: cps/templates/config_db.html:80 msgid "New db location is invalid, please enter valid path" -msgstr "A localización da base de datos non é válida. Por favor, Introduce a ruta correcta" +msgstr "A nova localización da base de datos non é válida, introduce unha ruta válida" #: cps/templates/config_edit.html:18 msgid "Server Configuration" @@ -2199,11 +2235,11 @@ msgstr "Porto do servidor" #: cps/templates/config_edit.html:28 msgid "SSL certfile location (leave it empty for non-SSL Servers)" -msgstr "Localización do arquivo de certificado SSL (deixar en branco se non hai un servidor SSL)" +msgstr "Localización do ficheiro de certificado SSL (déixao baleiro para servidores non SSL)" #: cps/templates/config_edit.html:35 msgid "SSL Keyfile location (leave it empty for non-SSL Servers)" -msgstr "Localización do arquivo clave SSL (dejar en blanco si no hay un servidor SSL)" +msgstr "Localización do ficheiro de chave SSL (déixao baleiro para servidores non SSL)" #: cps/templates/config_edit.html:43 msgid "Update Channel" @@ -2215,67 +2251,67 @@ msgstr "Estable" #: cps/templates/config_edit.html:46 msgid "Nightly" -msgstr "Nocturno" +msgstr "Noite" #: cps/templates/config_edit.html:50 msgid "Trusted Hosts (Comma Separated)" -msgstr "Anfitrións de confianza (Separados por comas)" +msgstr "Hosts de confianza (separados por comas)" #: cps/templates/config_edit.html:61 msgid "Logfile Configuration" -msgstr "Configuración do arquivo de rexistro" +msgstr "Configuración do ficheiro de rexistro" #: cps/templates/config_edit.html:77 msgid "Location and name of logfile (calibre-web.log for no entry)" -msgstr "Localización e nome do arquivo de rexistro (se non se especifica será calibre-web.log)" +msgstr "Localización e nome do ficheiro de rexistro (calibre-web.log sen entrada)" #: cps/templates/config_edit.html:82 msgid "Enable Access Log" -msgstr "Habilitar rexistro de acceso" +msgstr "Activar o rexistro de acceso" #: cps/templates/config_edit.html:85 msgid "Location and name of access logfile (access.log for no entry)" -msgstr "Localización e nombre do archivo de rexistro de acceso (access.log non ten ningunha entrada)" +msgstr "Localización e nome do ficheiro de rexistro de acceso (access.log sen entrada)" #: cps/templates/config_edit.html:96 msgid "Feature Configuration" -msgstr "Configuración de características" +msgstr "Configuración de funcións" #: cps/templates/config_edit.html:104 msgid "Convert non-English characters in title and author while saving to disk" -msgstr "Convertir caracteres non ingleses no título e no autor mentres se graba no disco" +msgstr "Converte caracteres non ingleses en título e autor mentres gardas no disco" #: cps/templates/config_edit.html:108 msgid "Embed Metadata to Ebook File on Download/Conversion/e-mail (needs Calibre/Kepubify binaries)" -msgstr "" +msgstr "Inserir metadatos no ficheiro de libro electrónico na descarga/conversión/correo electrónico (necesita os binarios de Calibre/Kepubify)" #: cps/templates/config_edit.html:112 msgid "Enable Uploads" -msgstr "Permitir cargas" +msgstr "Activa as cargas" #: cps/templates/config_edit.html:112 msgid "(Please ensure that users also have upload permissions)" -msgstr "(Por favor asegúrese que os usuarios teñen permisos de carga)" +msgstr "(Asegúrate de que os usuarios tamén teñan permisos de carga)" #: cps/templates/config_edit.html:116 msgid "Allowed Upload Fileformats" -msgstr "Formatos de arquivo permitidos para carga" +msgstr "Formatos de ficheiro de carga permitidos" #: cps/templates/config_edit.html:122 msgid "Enable Anonymous Browsing" -msgstr "Permitir navegación anónima" +msgstr "Activar a navegación anónima" #: cps/templates/config_edit.html:126 msgid "Enable Public Registration" -msgstr "Permitir rexistro público" +msgstr "Activar o rexistro público" #: cps/templates/config_edit.html:131 msgid "Use Email as Username" -msgstr "Utilizar correo electrónico como nome de usuario" +msgstr "Use o correo electrónico como nome de usuario" #: cps/templates/config_edit.html:136 msgid "Enable Magic Link Remote Login" -msgstr "Permitir inicio de sesión remoto (\"magic link\")" +msgstr "Activa o inicio de sesión remoto de Magic Link" #: cps/templates/config_edit.html:141 msgid "Enable Kobo sync" @@ -2287,19 +2323,19 @@ msgstr "Peticións proxy á tenda Kobo descoñecidas" #: cps/templates/config_edit.html:149 msgid "Server External Port (for port forwarded API calls)" -msgstr "Porto externo do servidor (para peticións API)" +msgstr "Porto externo do servidor (para chamadas á API de reenvío de portos)" #: cps/templates/config_edit.html:157 msgid "Use Goodreads" -msgstr "Usar Goodreads" +msgstr "Use Goodreads" #: cps/templates/config_edit.html:161 msgid "Goodreads API Key" -msgstr "Goodreads API Key" +msgstr "Clave da API de Goodreads" #: cps/templates/config_edit.html:168 msgid "Allow Reverse Proxy Authentication" -msgstr "Permitir Autenticación Proxy Inversa" +msgstr "Permitir a autenticación de proxy inverso" #: cps/templates/config_edit.html:179 msgid "Login type" @@ -2307,19 +2343,19 @@ msgstr "Tipo de inicio de sesión" #: cps/templates/config_edit.html:181 msgid "Use Standard Authentication" -msgstr "Usar autenticación estándar" +msgstr "Use a autenticación estándar" #: cps/templates/config_edit.html:183 msgid "Use LDAP Authentication" -msgstr "Usar autenticación LDAP" +msgstr "Use a autenticación LDAP" #: cps/templates/config_edit.html:186 msgid "Use OAuth" -msgstr "Usar OAuth" +msgstr "Usa OAuth" #: cps/templates/config_edit.html:193 msgid "LDAP Server Host Name or IP Address" -msgstr "Nome de anfitrión ou enderezo IP do servidor LDAP" +msgstr "Nome de host do servidor LDAP ou enderezo IP" #: cps/templates/config_edit.html:197 msgid "LDAP Server Port" @@ -2339,15 +2375,15 @@ msgstr "SSL" #: cps/templates/config_edit.html:209 msgid "LDAP CACertificate Path (Only needed for Client Certificate Authentication)" -msgstr "Ruta LDAP CACertificate (So necesaria para certificado de autenticación de cliente)" +msgstr "Ruta do certificado LDAP da autoridade certificadora (só necesario para a autenticación do certificado do cliente)" #: cps/templates/config_edit.html:216 msgid "LDAP Certificate Path (Only needed for Client Certificate Authentication)" -msgstr "Ruta LDAP Certificate (So necesaria para certificado de autenticación de cliente)" +msgstr "Ruta do certificado LDAP (só necesario para a autenticación do certificado do cliente)" #: cps/templates/config_edit.html:223 msgid "LDAP Keyfile Path (Only needed for Client Certificate Authentication)" -msgstr "Ruta LDAP Keyfile (So necesaria para certificado de autenticación de cliente)" +msgstr "Ruta do ficheiro de chave LDAP (só necesario para a autenticación do certificado do cliente)" #: cps/templates/config_edit.html:232 msgid "LDAP Authentication" @@ -2367,7 +2403,7 @@ msgstr "Simple" #: cps/templates/config_edit.html:241 msgid "LDAP Administrator Username" -msgstr "Nome de usuario de administrador LDAP" +msgstr "Nome de usuario do administrador LDAP" #: cps/templates/config_edit.html:247 msgid "LDAP Administrator Password" @@ -2387,7 +2423,7 @@ msgstr "O servidor LDAP é OpenLDAP?" #: cps/templates/config_edit.html:263 msgid "Following Settings are Needed For User Import" -msgstr "As seguintes configuracións son necesarias para a importación de usuarios" +msgstr "Os seguintes axustes son necesarios para a importación do usuario" #: cps/templates/config_edit.html:265 msgid "LDAP Group Object Filter" @@ -2395,19 +2431,19 @@ msgstr "Filtro de obxectos de grupo LDAP" #: cps/templates/config_edit.html:269 msgid "LDAP Group Name" -msgstr "Nome de grupo LDAP" +msgstr "Nome do grupo LDAP" #: cps/templates/config_edit.html:273 msgid "LDAP Group Members Field" -msgstr "Campo de membros de grupo LDAP" +msgstr "Campo de membros do grupo LDAP" #: cps/templates/config_edit.html:277 msgid "LDAP Member User Filter Detection" -msgstr "Filtro de detección LDAP Member User" +msgstr "Detección de filtros de usuarios de membros de LDAP" #: cps/templates/config_edit.html:279 msgid "Autodetect" -msgstr "Auto detectar" +msgstr "Autodetección" #: cps/templates/config_edit.html:280 msgid "Custom Filter" @@ -2415,106 +2451,102 @@ msgstr "Filtro personalizado" #: cps/templates/config_edit.html:285 msgid "LDAP Member User Filter" -msgstr "Filtro LDAP Member User" +msgstr "Filtro de usuarios de membros de LDAP" #: cps/templates/config_edit.html:296 #, python-format msgid "Obtain %(provider)s OAuth Credential" -msgstr "Obter a Credencial OAuth de %(provider)s" +msgstr "Obter a credencial OAuth de %(provider)s" #: cps/templates/config_edit.html:299 #, python-format msgid "%(provider)s OAuth Client Id" -msgstr "Id de cliente de OAuth de %(provider)s" +msgstr "Id. de cliente OAuth de %(provider)s" #: cps/templates/config_edit.html:303 #, python-format msgid "%(provider)s OAuth Client Secret" -msgstr "Secreto OAuth de Cliente de %(provider)s" +msgstr "Segredo de cliente OAuth de %(provider)s" #: cps/templates/config_edit.html:319 msgid "External binaries" msgstr "Binarios externos" #: cps/templates/config_edit.html:325 -#, fuzzy msgid "Path to Calibre Binaries" -msgstr "Ruta para Calibre E-Book Converter" +msgstr "Camiño aos binarios de Calibre" #: cps/templates/config_edit.html:333 msgid "Calibre E-Book Converter Settings" -msgstr "Axustes de Calibre E-Book Converter" +msgstr "Configuración do conversor de libros electrónicos de Calibre" #: cps/templates/config_edit.html:336 msgid "Path to Kepubify E-Book Converter" -msgstr "Ruta para Kepubify E-Book Converter" +msgstr "Camiño ao conversor de libros electrónicos Kepubify" #: cps/templates/config_edit.html:344 -#, fuzzy msgid "Location of Unrar binary" -msgstr "Localización do binario de UnRar" +msgstr "Localización do binario Unrar" #: cps/templates/config_edit.html:360 -#, fuzzy msgid "Security Settings" -msgstr "Axustes OAuth" +msgstr "Configuración de seguranza" #: cps/templates/config_edit.html:368 msgid "Limit failed login attempts" -msgstr "" +msgstr "Limite os intentos de inicio de sesión errados" #: cps/templates/config_edit.html:372 msgid "Configure Backend for Limiter" -msgstr "" +msgstr "Configurar o backend para o limitador" #: cps/templates/config_edit.html:376 msgid "Options for Limiter Backend" -msgstr "" +msgstr "Opcións para o backend do limitador" #: cps/templates/config_edit.html:382 msgid "Check if file extensions matches file content on upload" -msgstr "" +msgstr "Comproba se as extensións do ficheiro coinciden co contido do ficheiro na carga" #: cps/templates/config_edit.html:385 msgid "Session protection" -msgstr "" +msgstr "Protección de sesión" #: cps/templates/config_edit.html:387 msgid "Basic" -msgstr "" +msgstr "Básico" #: cps/templates/config_edit.html:388 msgid "Strong" -msgstr "" +msgstr "Forte" #: cps/templates/config_edit.html:393 -#, fuzzy msgid "User Password policy" -msgstr "Restablecer contrasinal de usuario" +msgstr "Política de contrasinais de usuario" #: cps/templates/config_edit.html:397 msgid "Minimum password length" -msgstr "" +msgstr "Lonxitude mínima do contrasinal" #: cps/templates/config_edit.html:402 msgid "Enforce number" -msgstr "" +msgstr "Aplicar número" #: cps/templates/config_edit.html:406 msgid "Enforce lowercase characters" -msgstr "" +msgstr "Aplicar caracteres minúsculas" #: cps/templates/config_edit.html:410 msgid "Enforce uppercase characters" -msgstr "" +msgstr "Aplica os caracteres en maiúscula" #: cps/templates/config_edit.html:414 msgid "Enforce characters (needed For Chinese/Japanese/Korean Characters)" -msgstr "" +msgstr "Aplicar caracteres (necesario para caracteres chineses/xaponeses/coreanos)" #: cps/templates/config_edit.html:418 msgid "Enforce special characters" -msgstr "" +msgstr "Aplicar caracteres especiais" #: cps/templates/config_view_edit.html:17 msgid "View Configuration" @@ -2522,11 +2554,11 @@ msgstr "Ver configuración" #: cps/templates/config_view_edit.html:32 msgid "No. of Random Books to Display" -msgstr "Número de libros aleatorios a mostrar" +msgstr "Número de libros aleatorios para mostrar" #: cps/templates/config_view_edit.html:36 msgid "No. of Authors to Display Before Hiding (0=Disable Hiding)" -msgstr "Número de autores para mostrar antes de agochar (0 = desactivar o agochamento)" +msgstr "Número de autores a mostrar antes de ocultar (0=Desactivar ocultar)" #: cps/templates/config_view_edit.html:40 cps/templates/readcbr.html:101 msgid "Theme" @@ -2538,7 +2570,7 @@ msgstr "Tema estándar" #: cps/templates/config_view_edit.html:43 msgid "caliBlur! Dark Theme" -msgstr "caliBlur! Tema Escuro" +msgstr "caliBlur! Tema escuro" #: cps/templates/config_view_edit.html:47 msgid "Regular Expression for Ignoring Columns" @@ -2546,15 +2578,15 @@ msgstr "Expresión regular para ignorar columnas" #: cps/templates/config_view_edit.html:51 msgid "Link Read/Unread Status to Calibre Column" -msgstr "Enlazar á columna de Calibre de lido/sen ler" +msgstr "Enlace o estado de lectura/non lida á columna Calibre" #: cps/templates/config_view_edit.html:60 msgid "View Restrictions based on Calibre column" -msgstr "Ver restricións baseadas na columna de Calibre" +msgstr "Ver restricións baseadas na columna Calibre" #: cps/templates/config_view_edit.html:69 msgid "Regular Expression for Title Sorting" -msgstr "Expresión regular para ordear títulos" +msgstr "Expresión regular para a clasificación de títulos" #: cps/templates/config_view_edit.html:80 msgid "Default Settings for New Users" @@ -2570,11 +2602,11 @@ msgstr "Permitir descargas" #: cps/templates/config_view_edit.html:96 cps/templates/user_edit.html:105 msgid "Allow eBook Viewer" -msgstr "Permitir visor de libros" +msgstr "Permitir o visor de libros electrónicos" #: cps/templates/config_view_edit.html:101 cps/templates/user_edit.html:110 msgid "Allow Uploads" -msgstr "Permitir cargas de arquivos" +msgstr "Permitir cargas" #: cps/templates/config_view_edit.html:106 cps/templates/user_edit.html:115 msgid "Allow Edit" @@ -2582,27 +2614,27 @@ msgstr "Permitir editar" #: cps/templates/config_view_edit.html:111 cps/templates/user_edit.html:120 msgid "Allow Delete Books" -msgstr "Permitir borrar libros" +msgstr "Permitir eliminar libros" #: cps/templates/config_view_edit.html:116 cps/templates/user_edit.html:126 msgid "Allow Changing Password" -msgstr "Permitir cambiar a contrasinal" +msgstr "Permitir cambiar o contrasinal" #: cps/templates/config_view_edit.html:120 cps/templates/user_edit.html:130 msgid "Allow Editing Public Shelves" -msgstr "Permitir editar andeis públicos" +msgstr "Permitir editar estantes públicos" #: cps/templates/config_view_edit.html:123 msgid "Default Language" -msgstr "Lingua predeterminada" +msgstr "Idioma predeterminado" #: cps/templates/config_view_edit.html:131 msgid "Default Visible Language of Books" -msgstr "Lingua visible predeterminada dos libros" +msgstr "Linguaxe visible predeterminada dos libros" #: cps/templates/config_view_edit.html:147 msgid "Default Visibilities for New Users" -msgstr "Visibilidade predeterminada para novos usuarios" +msgstr "Visibilidades predeterminadas para novos usuarios" #: cps/templates/config_view_edit.html:163 cps/templates/user_edit.html:84 #: cps/templates/user_table.html:154 @@ -2611,104 +2643,87 @@ msgstr "Mostrar libros aleatorios na vista detallada" #: cps/templates/config_view_edit.html:166 cps/templates/user_edit.html:87 msgid "Add Allowed/Denied Tags" -msgstr "Engadir etiquetas Permitidas/denegados" +msgstr "Engadir etiquetas permitidas/denegadas" #: cps/templates/config_view_edit.html:167 msgid "Add Allowed/Denied custom column values" -msgstr "Engadir valores personalizados Permitidos/denegados" +msgstr "Engadir valores de columnas personalizados permitidos/denegados" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Ler no navegador" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" -msgstr "Escoitar no navegador" +msgstr "Escoita no navegador" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Libro %(index)s de %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publicado" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Marcar como non lido" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Marcar como lido" -#: cps/templates/detail.html:254 -#, fuzzy +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" -msgstr "Marcar como non lido" +msgstr "Marcar o libro como lido ou non lido" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" -msgstr "Lido" +msgstr "Ler" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" -msgstr "Restaurar dende o arquivo" +msgstr "Restaurar desde o arquivo" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" -msgstr "Engadir a arquivo" +msgstr "Engadir ao arquivo" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" -msgstr "" +msgstr "Marca o libro como arquivado ou non, para ocultalo en Calibre-Web e elimínao de Kobo Reader" -#: cps/templates/detail.html:267 -#, fuzzy +#: cps/templates/detail.html:275 msgid "Archive" -msgstr "Arquivado" +msgstr "Arquivo" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Descrición:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" -msgstr "Engadir ao andel" +msgstr "Engadir ao estante" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Público)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Editar metadatos" #: cps/templates/email_edit.html:13 msgid "Email Account Type" -msgstr "Escolle tipo de servidor" +msgstr "Tipo de conta de correo electrónico" #: cps/templates/email_edit.html:15 -#, fuzzy msgid "Standard Email Account" -msgstr "Usar conta de correo electrónico estándar" +msgstr "Conta de correo electrónico estándar" #: cps/templates/email_edit.html:16 -#, fuzzy msgid "Gmail Account" -msgstr "Escolle tipo de servidor" +msgstr "Conta de Gmail" #: cps/templates/email_edit.html:22 msgid "Setup Gmail Account" -msgstr "" +msgstr "Configurar a conta de Gmail" #: cps/templates/email_edit.html:24 msgid "Revoke Gmail Access" -msgstr "Revogar acceso a Gmail" +msgstr "Revogar o acceso a Gmail" #: cps/templates/email_edit.html:42 msgid "STARTTLS" @@ -2724,21 +2739,20 @@ msgstr "Contrasinal SMTP" #: cps/templates/email_edit.html:58 msgid "Attachment Size Limit" -msgstr "Tamaño límite do arquivo achegado" +msgstr "Límite de tamaño do anexo" #: cps/templates/email_edit.html:66 -#, fuzzy msgid "Save and Send Test Email" -msgstr "Gardar e enviar un correo electrónico de proba" +msgstr "Garda e envía correo electrónico de proba" #: cps/templates/email_edit.html:70 cps/templates/layout.html:26 #: cps/templates/shelf_order.html:42 cps/templates/user_table.html:174 msgid "Back" -msgstr "Voltar" +msgstr "De volta" #: cps/templates/email_edit.html:74 msgid "Allowed Domains (Whitelist)" -msgstr "Dominios permitidos para rexistrarse" +msgstr "Dominios permitidos (lista branca)" #: cps/templates/email_edit.html:78 cps/templates/email_edit.html:105 msgid "Add Domain" @@ -2751,23 +2765,19 @@ msgstr "Engadir" #: cps/templates/email_edit.html:86 cps/templates/email_edit.html:96 msgid "Enter domainname" -msgstr "Introducir nome de dominio" +msgstr "Introduza o nome de dominio" #: cps/templates/email_edit.html:92 msgid "Denied Domains (Blacklist)" -msgstr "Dominios prohibidos (Blaclist)" - -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Seguinte" +msgstr "Dominios denegados (lista negra)" #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" -msgstr "Abre o arquivo .kobo/Kobo/Kobo eReader.conf nun editor de texto e engade (ou edita):" +msgstr "Abra o ficheiro .kobo/Kobo/Kobo eReader.conf nun editor de texto e engada (ou edite):" #: cps/templates/generate_kobo_auth_url.html:11 msgid "Kobo Token:" -msgstr "Token de sincronización de Kobo" +msgstr "Kobo Token:" #: cps/templates/grid.html:21 msgid "List" @@ -2775,65 +2785,69 @@ msgstr "Lista" #: cps/templates/http_error.html:34 msgid "Calibre-Web Instance is unconfigured, please contact your administrator" -msgstr "A instancia de Calibre-Web non está configurada, por favor contacta co teu administrador" +msgstr "A instancia de Calibre-Web non está configurada, póñase en contacto co seu administrador" #: cps/templates/http_error.html:44 msgid "Create Issue" -msgstr "Crear una incidencia" +msgstr "Crear problema" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +msgid "Return to Database config" +msgstr "Volver á configuración da base de datos" + +#: cps/templates/http_error.html:54 msgid "Return to Home" -msgstr "Voltar ao inicio" +msgstr "Volver ao inicio" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" -msgstr "Pechar sesión" +msgstr "Saír Usuario" #: cps/templates/index.html:71 msgid "Sort ascending according to download count" -msgstr "Ordear carra arriba segundo a conta de descargas" +msgstr "Ordenar ascendente segundo o reconto de descargas" #: cps/templates/index.html:72 msgid "Sort descending according to download count" -msgstr "Ordear cara abaixo segundo a conta de descargas" +msgstr "Ordena descendente segundo o reconto de descargas" #: cps/templates/index.html:78 cps/templates/search.html:35 #: cps/templates/shelf.html:24 msgid "Sort authors in alphabetical order" -msgstr "Ordenar autores en orde alfabética" +msgstr "Ordenar os autores por orde alfabética" #: cps/templates/index.html:79 cps/templates/search.html:36 #: cps/templates/shelf.html:25 msgid "Sort authors in reverse alphabetical order" -msgstr "Ordenar autores en orde alfabética reversa" +msgstr "Ordenar os autores en orde alfabética inversa" #: cps/templates/index.html:83 msgid "Sort ascending according to series index" -msgstr "Ordear cara arriba segundo ao índice de serie" +msgstr "Ordena ascendente segundo o índice de serie" #: cps/templates/index.html:84 msgid "Sort descending according to series index" -msgstr "Ordear cara abaixo segundo ao índice de serie" +msgstr "Ordena descendente segundo o índice de serie" #: cps/templates/index.xml:7 msgid "Start" -msgstr "Comezar" +msgstr "Comeza" #: cps/templates/index.xml:19 msgid "Alphabetical Books" -msgstr "Libros alfabéticos" +msgstr "Libros Alfabéticos" #: cps/templates/index.xml:23 msgid "Books sorted alphabetically" -msgstr "Libros ordeados alfabéticamente" +msgstr "Libros ordenados alfabeticamente" #: cps/templates/index.xml:31 msgid "Popular publications from this catalog based on Downloads." -msgstr "Publicacións populares do catálogo baseadas nas descargas." +msgstr "Publicacións populares deste catálogo baseadas en Descargas." #: cps/templates/index.xml:40 msgid "Popular publications from this catalog based on Rating." -msgstr "Publicacións populares do catálogo baseadas na valoración." +msgstr "Publicacións populares deste catálogo baseadas na valoración." #: cps/templates/index.xml:45 msgid "Recently added Books" @@ -2841,103 +2855,80 @@ msgstr "Libros engadidos recentemente" #: cps/templates/index.xml:49 msgid "The latest Books" -msgstr "Últimos libros" +msgstr "Os últimos Libros" #: cps/templates/index.xml:54 msgid "Random Books" -msgstr "Libros ao chou" +msgstr "Libros aleatorios" #: cps/templates/index.xml:83 msgid "Books ordered by Author" -msgstr "Libros ordeados por autor" +msgstr "Libros ordenados por autor" #: cps/templates/index.xml:92 msgid "Books ordered by publisher" -msgstr "Libros ordeados por editor" +msgstr "Libros ordenados pola editorial" #: cps/templates/index.xml:101 msgid "Books ordered by category" -msgstr "Libros ordeados por categorías" +msgstr "Libros ordenados por categoría" #: cps/templates/index.xml:110 msgid "Books ordered by series" -msgstr "Libros ordeados por series" +msgstr "Libros ordenados por series" #: cps/templates/index.xml:119 msgid "Books ordered by Languages" -msgstr "Libros ordeados por lingua" +msgstr "Libros ordenados por Linguas" #: cps/templates/index.xml:128 msgid "Books ordered by Rating" -msgstr "Libros ordeados por valoración" +msgstr "Libros ordenados por valoración" #: cps/templates/index.xml:137 msgid "Books ordered by file formats" -msgstr "Libros ordeados por formato de arquivo" +msgstr "Libros ordenados por formatos de arquivo" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" -msgstr "Andeis" +msgstr "Estantes" #: cps/templates/index.xml:146 msgid "Books organized in shelves" -msgstr "Libros organizados en andeis" - -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Inicio" +msgstr "Libros organizados en estantes" #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Alternar navegación" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Buscar na librería" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Simple" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Conta" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Pechar sesión" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Cargando..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Erro" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Carga feita, procesando, por favor agarde..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" -msgstr "Axustes" +msgstr "Configuración" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" -msgstr "Por favor, non actualice a páxina" +msgstr "Non actualices a páxina" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" -msgstr "Navegar" +msgstr "Explorar" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" -msgstr "Acerca de" +msgstr "Sobre" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Previo" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Detalles do libro" @@ -2951,19 +2942,19 @@ msgstr "Arquivado" #: cps/templates/login.html:18 msgid "Remember Me" -msgstr "Lembrarme" +msgstr "Lémbrame" #: cps/templates/login.html:23 msgid "Forgot Password?" -msgstr "Esqueceu a contrasinal?" +msgstr "Esqueceches o contrasinal?" #: cps/templates/login.html:34 msgid "Log in with Magic Link" -msgstr "Iniciar sesión cun enlace máxico" +msgstr "Inicia sesión con Magic Link" #: cps/templates/logviewer.html:6 msgid "Show Calibre-Web Log: " -msgstr "Mostrar o rexistro de Calibre-Web: " +msgstr "Mostrar rexistro de Calibre-Web: " #: cps/templates/logviewer.html:8 msgid "Calibre-Web Log: " @@ -2971,35 +2962,35 @@ msgstr "Rexistro de Calibre-Web: " #: cps/templates/logviewer.html:8 msgid "Stream output, can't be displayed" -msgstr "A saída do fluxo non se pode mostrar" +msgstr "Saída da emisión, non se pode mostrar" #: cps/templates/logviewer.html:12 msgid "Show Access Log: " -msgstr "Mostrar rexistro de acceso:" +msgstr "Mostrar rexistro de acceso: " #: cps/templates/logviewer.html:18 msgid "Download Calibre-Web Log" -msgstr "Descargar o rexistro de Calibre-Web" +msgstr "Descargar rexistro de Calibre-Web" #: cps/templates/logviewer.html:21 msgid "Download Access Log" -msgstr "Descargar o rexistro de acceso" +msgstr "Descargar rexistro de acceso" #: cps/templates/modal_dialogs.html:6 msgid "Select Allowed/Denied Tags" -msgstr "Seleccionar etiquetas Permitidas/Negadas" +msgstr "Seleccione Etiquetas permitidas/denegadas" #: cps/templates/modal_dialogs.html:7 msgid "Select Allowed/Denied Custom Column Values" -msgstr "Seleccionar valores propios de columnas Permitidas/Denegadas" +msgstr "Seleccione Valores de columna personalizados permitidos/denegados" #: cps/templates/modal_dialogs.html:8 msgid "Select Allowed/Denied Tags of User" -msgstr "Seleccionar etiquetas de usuario Permitido/Denegado" +msgstr "Seleccione Etiquetas permitidas/denegadas do usuario" #: cps/templates/modal_dialogs.html:9 msgid "Select Allowed/Denied Custom Column Values of User" -msgstr "Seleccionar columnas personalizadas de usuario Permitido/Denegado" +msgstr "Seleccione Valores de columna personalizados permitidos/denegados do usuario" #: cps/templates/modal_dialogs.html:15 msgid "Enter Tag" @@ -3007,7 +2998,7 @@ msgstr "Introduce a etiqueta" #: cps/templates/modal_dialogs.html:24 msgid "Add View Restriction" -msgstr "Engadir restricción de vista" +msgstr "Engadir restrición de visualización" #: cps/templates/modal_dialogs.html:50 msgid "This book format will be permanently erased from database" @@ -3015,23 +3006,23 @@ msgstr "Este formato de libro borrarase permanentemente da base de datos" #: cps/templates/modal_dialogs.html:51 msgid "This book will be permanently erased from database" -msgstr "O libro borrarase permanentemente da base de datos" +msgstr "Este libro borrarase permanentemente da base de datos" #: cps/templates/modal_dialogs.html:52 msgid "and hard disk" -msgstr "e do disco duro" +msgstr "e disco duro" #: cps/templates/modal_dialogs.html:56 msgid "Important Kobo Note: deleted books will remain on any paired Kobo device." -msgstr "Nota de Kobo importante: os libros eliminados permanecerán nos dispositivos Kobo emparellados." +msgstr "Nota importante de Kobo: os libros eliminados permanecerán en calquera dispositivo Kobo vinculado." #: cps/templates/modal_dialogs.html:57 msgid "Books must first be archived and the device synced before a book can safely be deleted." -msgstr "Antes de que un libro poida eliminarse con seguridade debe arquivarse e sincronizarse co dispositivo." +msgstr "Primeiro hai que arquivar os libros e sincronizar o dispositivo para que se poida eliminar un libro con seguridade." #: cps/templates/modal_dialogs.html:76 msgid "Choose File Location" -msgstr "Escolla a localización do arquivo" +msgstr "Escolla a localización do ficheiro" #: cps/templates/modal_dialogs.html:82 msgid "type" @@ -3051,47 +3042,87 @@ msgstr "Directorio pai" #: cps/templates/modal_dialogs.html:98 msgid "Select" -msgstr "Seleccionar" +msgstr "Seleccione" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Vale" #: cps/templates/osd.xml:5 msgid "Calibre-Web eBook Catalog" -msgstr "Catálogo de libros electrónicos de Calibre-Web" +msgstr "Catálogo de libros electrónicos Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" -msgstr "Lector epub" +msgstr "lector epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +msgid "Choose a theme below:" +msgstr "Escolle un tema a continuación:" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Claro" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" -msgstr "Oscuro" +msgstr "Escuro" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sepia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Negro" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." -msgstr "Refluxo do texto cando as barras laterais están abertas." +msgstr "Redistribuír o texto cando as barras laterais estean abertas." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" -msgstr "" +msgstr "Tamaños de letra" + +#: cps/templates/read.html:105 +msgid "Font" +msgstr "Fonte" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "Por defecto" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "Yahei" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "SimSun" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "KaiTi" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "Espallamento" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "Dúas columnas" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "Unha columna" #: cps/templates/readcbr.html:8 msgid "Comic Reader" -msgstr "Lector de Cómics" +msgstr "Lector de cómics" #: cps/templates/readcbr.html:75 msgid "Keyboard Shortcuts" @@ -3107,68 +3138,67 @@ msgstr "Páxina seguinte" #: cps/templates/readcbr.html:80 msgid "Single Page Display" -msgstr "" +msgstr "Visualización dunha soa páxina" #: cps/templates/readcbr.html:81 msgid "Long Strip Display" -msgstr "" +msgstr "Pantalla de tira longa" #: cps/templates/readcbr.html:82 msgid "Scale to Best" -msgstr "Escalar ao mellor" +msgstr "Escala ao mellor" #: cps/templates/readcbr.html:83 msgid "Scale to Width" -msgstr "Escalar ao ancho" +msgstr "Escala ao ancho" #: cps/templates/readcbr.html:84 msgid "Scale to Height" -msgstr "Escalar ao alto" +msgstr "Escala á altura" #: cps/templates/readcbr.html:85 msgid "Scale to Native" -msgstr "Escalado nativo" +msgstr "Escalar a nativo" #: cps/templates/readcbr.html:86 msgid "Rotate Right" -msgstr "Xirar cara a dereita" +msgstr "Xirar á dereita" #: cps/templates/readcbr.html:87 msgid "Rotate Left" -msgstr "Xirar cara a esquerda" +msgstr "Xirar á esquerda" #: cps/templates/readcbr.html:88 msgid "Flip Image" -msgstr "Voltear a imaxe" +msgstr "Voltar imaxe" #: cps/templates/readcbr.html:110 msgid "Display" -msgstr "" +msgstr "Mostrar" #: cps/templates/readcbr.html:113 -#, fuzzy msgid "Single Page" -msgstr "Páxina de administración" +msgstr "Páxina única" #: cps/templates/readcbr.html:114 msgid "Long Strip" -msgstr "" +msgstr "Tira longa" #: cps/templates/readcbr.html:119 msgid "Scale" -msgstr "Escalar" +msgstr "Escala" #: cps/templates/readcbr.html:122 msgid "Best" -msgstr "Mellor" +msgstr "O mellor" #: cps/templates/readcbr.html:123 msgid "Width" -msgstr "Ancho" +msgstr "Anchura" #: cps/templates/readcbr.html:124 msgid "Height" -msgstr "Alto" +msgstr "Altura" #: cps/templates/readcbr.html:125 msgid "Native" @@ -3176,11 +3206,11 @@ msgstr "Nativo" #: cps/templates/readcbr.html:130 msgid "Rotate" -msgstr "Rotar" +msgstr "Xirar" #: cps/templates/readcbr.html:141 msgid "Flip" -msgstr "Voltear" +msgstr "Voltar" #: cps/templates/readcbr.html:144 msgid "Horizontal" @@ -3204,11 +3234,11 @@ msgstr "De dereita a esquerda" #: cps/templates/readcbr.html:162 msgid "Reset to Top" -msgstr "" +msgstr "Restablecer a parte superior" #: cps/templates/readcbr.html:163 msgid "Remember Position" -msgstr "" +msgstr "Lembrar a posición" #: cps/templates/readcbr.html:168 msgid "Scrollbar" @@ -3220,7 +3250,7 @@ msgstr "Mostrar" #: cps/templates/readcbr.html:172 msgid "Hide" -msgstr "Agochar" +msgstr "Ocultar" #: cps/templates/readdjvu.html:5 msgid "DJVU Reader" @@ -3228,15 +3258,15 @@ msgstr "Lector DJVU" #: cps/templates/readpdf.html:31 msgid "PDF Reader" -msgstr "Lector PDF" +msgstr "Lector de PDF" #: cps/templates/readtxt.html:6 msgid "txt Reader" -msgstr "Lector txt" +msgstr "lector txt" #: cps/templates/register.html:4 msgid "Register New Account" -msgstr "Rexistre unha conta nova" +msgstr "Rexistrar unha nova conta" #: cps/templates/register.html:10 msgid "Choose a username" @@ -3244,31 +3274,27 @@ msgstr "Escolla un nome de usuario" #: cps/templates/register.html:15 msgid "Your Email" -msgstr "O teu enderezo de correo" +msgstr "O teu correo electrónico" #: cps/templates/remote_login.html:5 msgid "Magic Link - Authorise New Device" -msgstr "Enlace Máxico - Autorizar un novo dispositivo" +msgstr "Magic Link - Autorizar un novo dispositivo" #: cps/templates/remote_login.html:7 msgid "On another device, login and visit:" -msgstr "Utiliza otro dispositivo, inicia sesión e visita:" +msgstr "Noutro dispositivo, inicia sesión e visita:" #: cps/templates/remote_login.html:11 msgid "Once verified, you will automatically be logged in on this device." -msgstr "Unha vez verificado, iniciará sesión automáticamente neste dispositivo." +msgstr "Unha vez verificado, iniciarase sesión automaticamente neste dispositivo." #: cps/templates/remote_login.html:14 msgid "This verification link will expire in 10 minutes." -msgstr "O enlace de verificación caducará despois de 10 minutos." +msgstr "Esta ligazón de verificación caducará dentro de 10 minutos." #: cps/templates/schedule_edit.html:33 msgid "Generate Series Cover Thumbnails" -msgstr "Xerar miniaturas de cubertas de series" - -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Non se atoparon resultados" +msgstr "Xerar miniaturas de portada de serie" #: cps/templates/search.html:7 msgid "Search Term:" @@ -3280,35 +3306,35 @@ msgstr "Resultados para:" #: cps/templates/search_form.html:21 msgid "Published Date From" -msgstr "Data de publicación dende" +msgstr "Data de publicación Dende" #: cps/templates/search_form.html:31 msgid "Published Date To" msgstr "Data de publicación ata" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" -msgstr "" +msgstr "Calquera" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" -msgstr "" +msgstr "Baleiro" #: cps/templates/search_form.html:60 msgid "Exclude Tags" -msgstr "Excluir etiquetas" +msgstr "Excluír etiquetas" #: cps/templates/search_form.html:78 msgid "Exclude Series" -msgstr "Excluir series" +msgstr "Excluír serie" #: cps/templates/search_form.html:96 msgid "Exclude Shelves" -msgstr "Excluir andeis" +msgstr "Excluír estantes" #: cps/templates/search_form.html:116 msgid "Exclude Languages" -msgstr "Excluir linguas" +msgstr "Excluír idiomas" #: cps/templates/search_form.html:127 msgid "Extensions" @@ -3316,7 +3342,7 @@ msgstr "Extensións" #: cps/templates/search_form.html:135 msgid "Exclude Extensions" -msgstr "Excluir extensións" +msgstr "Excluír extensións" #: cps/templates/search_form.html:145 msgid "Rating Above" @@ -3326,41 +3352,51 @@ msgstr "Valoración superior a" msgid "Rating Below" msgstr "Valoración inferior a" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "De:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Para:" #: cps/templates/shelf.html:13 msgid "Delete this Shelf" -msgstr "Borrar este andel" +msgstr "Eliminar este estante" #: cps/templates/shelf.html:14 msgid "Edit Shelf Properties" -msgstr "Editar propiedades do andel" +msgstr "Editar propiedades do estante" #: cps/templates/shelf.html:17 msgid "Arrange books manually" -msgstr "Ordear libros manualmente" +msgstr "Organiza os libros manualmente" #: cps/templates/shelf.html:18 msgid "Disable Change order" -msgstr "Desactivar o cambio de orde" +msgstr "Desactivar Cambiar orde" #: cps/templates/shelf.html:18 msgid "Enable Change order" -msgstr "Activar o cambio de orde" +msgstr "Activar Cambiar orde" + +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "Ordena segundo o libro engadido ao estante, primeiro o máis novo" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "Ordena segundo o libro engadido ao estante, primeiro o máis antigo" #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" -msgstr "Compartir con todos" +msgstr "Comparte con todos" #: cps/templates/shelf_edit.html:21 msgid "Sync this shelf with Kobo device" -msgstr "Sincronizar este andel cun dispositivo Kobo" +msgstr "Sincroniza este estante co dispositivo Kobo" #: cps/templates/shelf_order.html:5 msgid "Drag to Rearrange Order" @@ -3368,11 +3404,11 @@ msgstr "Arrastra para reorganizar a orde" #: cps/templates/shelf_order.html:33 msgid "Hidden Book" -msgstr "Libro agochado" +msgstr "Libro oculto" #: cps/templates/stats.html:7 msgid "Library Statistics" -msgstr "Estatísticas da Biblioteca" +msgstr "Estadística da biblioteca" #: cps/templates/stats.html:12 msgid "Books in this Library" @@ -3380,19 +3416,19 @@ msgstr "Libros nesta biblioteca" #: cps/templates/stats.html:16 msgid "Authors in this Library" -msgstr "Autores nesta biblioteca" +msgstr "Autores desta biblioteca" #: cps/templates/stats.html:20 msgid "Categories in this Library" -msgstr "Categorías nesta biblioteca" +msgstr "Categorías desta biblioteca" #: cps/templates/stats.html:24 msgid "Series in this Library" -msgstr "Series nesta biblioteca" +msgstr "Serie nesta biblioteca" #: cps/templates/stats.html:29 msgid "System Statistics" -msgstr "Estadísticas do sistema" +msgstr "Estatística do sistema" #: cps/templates/stats.html:33 msgid "Program" @@ -3422,41 +3458,49 @@ msgstr "Progreso" msgid "Run Time" msgstr "Tempo de execución" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "Mensaxe" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Accións" -#: cps/templates/tasks.html:40 -msgid "This task will be cancelled. Any progress made by this task will be saved." -msgstr "Esta tarefa cancelarase. Non se gardará ningún progreso feito por esta tarefa." - #: cps/templates/tasks.html:41 +msgid "This task will be cancelled. Any progress made by this task will be saved." +msgstr "Esta tarefa cancelarase. Calquera progreso realizado nesta tarefa gardarase." + +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." -msgstr "Se esta é unha tarefa programada, volverá a lanzarse na próxima hora programada." +msgstr "Se esta é unha tarefa programada, volverase executar durante a próxima hora programada." #: cps/templates/user_edit.html:20 msgid "Reset user Password" -msgstr "Restablecer contrasinal de usuario" +msgstr "Restablecer o contrasinal do usuario" + +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "Enviar ao enderezo de correo electrónico do eReader. Use comas para separar os correos electrónicos para varios eReaders" #: cps/templates/user_edit.html:43 msgid "Language of Books" -msgstr "Linguaxe de libros" +msgstr "Linguaxe dos libros" #: cps/templates/user_edit.html:54 msgid "OAuth Settings" -msgstr "Axustes OAuth" +msgstr "Configuración de OAuth" #: cps/templates/user_edit.html:56 msgid "Link" -msgstr "Ligar" +msgstr "Vincular" #: cps/templates/user_edit.html:58 msgid "Unlink" -msgstr "Desligar" +msgstr "Desvincular" #: cps/templates/user_edit.html:64 msgid "Kobo Sync Token" -msgstr "Token de sincronización de Kobo" +msgstr "Token de sincronización Kobo" #: cps/templates/user_edit.html:66 msgid "Create/View" @@ -3464,27 +3508,27 @@ msgstr "Crear/Ver" #: cps/templates/user_edit.html:70 msgid "Force full kobo sync" -msgstr "Forzar sincronización completa de kobo" +msgstr "Forzar a sincronización completa de kobo" #: cps/templates/user_edit.html:88 msgid "Add allowed/Denied Custom Column Values" -msgstr "Engadir columnas de valores propios de Permitidos/Denegados" +msgstr "Engadir valores de columna personalizados permitidos/denegados" #: cps/templates/user_edit.html:137 msgid "Sync only books in selected shelves with Kobo" -msgstr "Sincronizar con Kobo soamente os libros dos andeis seleccionados" +msgstr "Sincroniza só libros nos estantes seleccionados con Kobo" #: cps/templates/user_edit.html:147 cps/templates/user_table.html:169 msgid "Delete User" -msgstr "Borrar usuario" +msgstr "Eliminar usuario" #: cps/templates/user_edit.html:159 msgid "Generate Kobo Auth URL" -msgstr "Xerar Auth URL de Kobo" +msgstr "Xerar URL de autenticación de Kobo" #: cps/templates/user_table.html:80 cps/templates/user_table.html:103 msgid "Select..." -msgstr "Selección..." +msgstr "Selecciona..." #: cps/templates/user_table.html:131 msgid "Edit User" @@ -3492,26 +3536,23 @@ msgstr "Editar usuario" #: cps/templates/user_table.html:134 msgid "Enter Username" -msgstr "Introduce o nome do usuario" +msgstr "Introduza o nome de usuario" #: cps/templates/user_table.html:135 -#, fuzzy msgid "Enter Email" -msgstr "Comprobar correo electrónico" +msgstr "Introduce o correo electrónico" #: cps/templates/user_table.html:136 -#, fuzzy msgid "Enter eReader Email" -msgstr "Enviar ao enderezo de correo electrónico do Kindle" +msgstr "Introduza o correo electrónico do eReader" #: cps/templates/user_table.html:136 -#, fuzzy msgid "eReader Email" -msgstr "Comprobar o correo electrónico" +msgstr "correo electrónico do eReader" #: cps/templates/user_table.html:137 msgid "Locale" -msgstr "Lingua" +msgstr "Local" #: cps/templates/user_table.html:138 msgid "Visible Book Languages" @@ -3519,54 +3560,53 @@ msgstr "Linguas de libros visibles" #: cps/templates/user_table.html:139 msgid "Edit Allowed Tags" -msgstr "Editar etiquetas Permitidas" +msgstr "Editar etiquetas permitidas" #: cps/templates/user_table.html:139 msgid "Allowed Tags" -msgstr "Etiquetas Permitidas" +msgstr "Etiquetas permitidas" #: cps/templates/user_table.html:140 msgid "Edit Denied Tags" -msgstr "Editar Etiquetas Denegadas" +msgstr "Editar etiquetas denegadas" #: cps/templates/user_table.html:140 msgid "Denied Tags" -msgstr "Etiquetas Denegadas" +msgstr "Etiquetas denegadas" #: cps/templates/user_table.html:141 msgid "Edit Allowed Column Values" -msgstr "Editar valores permitidos para a columna" +msgstr "Editar valores de columna permitidos" #: cps/templates/user_table.html:141 msgid "Allowed Column Values" -msgstr "Valores permitidos da columna" +msgstr "Valores de columna permitidos" #: cps/templates/user_table.html:142 msgid "Edit Denied Column Values" -msgstr "Editar valores non permitidos para a columna" +msgstr "Editar valores de columna denegados" #: cps/templates/user_table.html:142 msgid "Denied Column Values" -msgstr "Valores non permitidos dea columna" +msgstr "Valores de columna denegados" #: cps/templates/user_table.html:144 msgid "Change Password" -msgstr "Cambiar a contrasinal" +msgstr "Cambiar contrasinal" #: cps/templates/user_table.html:147 msgid "View" -msgstr "Vista" +msgstr "Ver" #: cps/templates/user_table.html:150 msgid "Edit Public Shelves" -msgstr "Editar andeis públicos" +msgstr "Editar estanterías públicas" #: cps/templates/user_table.html:152 msgid "Sync selected Shelves with Kobo" -msgstr "sincronizar andeis seleccionados con Kobo" +msgstr "Sincroniza os estantes seleccionados con Kobo" #: cps/templates/user_table.html:156 -#, fuzzy msgid "Show Read/Unread Section" -msgstr "Mostrar selección lidos/non lidos" +msgstr "Mostrar a sección lida/non lida" diff --git a/cps/translations/hu/LC_MESSAGES/messages.mo b/cps/translations/hu/LC_MESSAGES/messages.mo index 26b9f23a..d584f454 100644 Binary files a/cps/translations/hu/LC_MESSAGES/messages.mo and b/cps/translations/hu/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/hu/LC_MESSAGES/messages.po b/cps/translations/hu/LC_MESSAGES/messages.po index c33e9601..4b32f85d 100644 --- a/cps/translations/hu/LC_MESSAGES/messages.po +++ b/cps/translations/hu/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: PROJECT VERSION\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2019-04-06 23:36+0200\n" "Last-Translator: \n" "Language: hu\n" @@ -16,508 +16,526 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statisztika" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "A kiszolgáló újraindult, tölts be újra az oldalt!" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "A kiszolgáló leállítása folyamatban, zárd be ezt az ablakot" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Ismeretlen" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Rendszergazda oldala" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Alapvető beállítások" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Felhasználói felület beállításai" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Rendszergazda felhasználó" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Mindent mutass" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "A Calibre-Web konfigurációja frissítve." -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Valóban törölni akarod a polcot?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Valóban törölni akarod a polcot?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Valóban törölni akarod a polcot?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Valóban törölni akarod a polcot?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Valóban le akarod állítani a Calibre-Web-et?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "SMTP beállítások változtatása" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Hiba történt a teszt levél küldése során: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Az e-mail kiszolgáló beállításai frissítve." -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Ismeretlen hiba történt. Próbáld újra később!" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr " A felhasználó szerkesztése: %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "A(z) %(user)s felhasználó jelszavának alaphelyzetbe állítása" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Először be kell állítani az SMTP levelező beállításokat..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Frissítési csomag kérése" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Frissítési csomag letöltése" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Frissítési csomag kitömörítése" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Fájlok cserélése" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Adatbázis kapcsolatok lezárva" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Szerver leállítása" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "A frissítés települt, kattints az OK-ra és újra tölt az oldal" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "A frissítés nem sikerült:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP hiba" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Kapcsolódási hiba" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Időtúllépés a kapcsolódás során" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Általános hiba" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Az e-mail kiszolgáló beállításai frissítve." -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Funkciók beállítása" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Az összes mezőt ki kell tölteni!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Az e-mail tartománya nem érvényes." -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Új felhasználó hozzáadása" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "A következő felhasználó létrehozva: %(user)s" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Már létezik felhasználó ehhez az e-mail címhez vagy felhasználói névhez." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "A felhasználó törölve: %(nick)s" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "A felhasználó frissítve: %(nick)s" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Keresés" + #: cps/converter.py:31 msgid "not installed" msgstr "nincs telepítve" @@ -526,128 +544,123 @@ msgstr "nincs telepítve" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Nincs" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Hiba történt az e-könyv megnyitásakor. A fájl nem létezik vagy nem érhető el:" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "A metaadatok sikeresen frissültek" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Az átalakításhoz hiányzik a forrás- vagy a célformátum!" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "A könyv sikeresen átalakításra lett jelölve a következő formátumra: %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Hiba történt a könyv átalakításakor: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Hiba történt az e-könyv megnyitásakor. A fájl nem létezik vagy nem érhető el:" + +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "A(z) %(langname)s nem érvényes nyelv" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "A metaadatok sikeresen frissültek" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "A(z) \"%(ext)s\" kiterjesztésű fájlok feltöltése nincs engedélyezve ezen a szerveren." -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "A(z) \"%(ext)s\" kiterjesztésű fájlok feltöltése nincs engedélyezve ezen a szerveren." -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "A feltöltendő fájlnak kiterjesztéssel kell rendelkeznie!" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "Metaadatok szerkesztése" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Nem sikerült létrehozni az elérési utat (engedély megtagadva): %(path)s." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Nem sikerült elmenteni a %(file)s fájlt." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "A(z) %(ext)s fájlformátum hozzáadva a könyvhez: %(book)s." @@ -660,482 +673,475 @@ msgstr "A Google Drive beállítása nem fejeződött be, próbáld kikapcsolni msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "A visszahívási tartomány nem ellenőrzött, kövesd az alábbi lépéseket a tartomány ellenőrzéséhez a Google Developer Console-ban:" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "A(z) %(format)s formátum nem található a következő könyvhöz: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s nem található a Google Drive-on: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s nem található: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Küldés Kindle-re" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Ez az e-mail a Calibre-Web-en keresztül lett küldve." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web teszt e-mail" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Teszt e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Kezdő lépések a Calibre-Web-bel" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Regisztrációs e-mail a következő felhasználóhoz: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "%(orig)s konvertálása %(format)s-ra és küldés Kindle-re" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "%(format)s küldése Kindle-re" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Küldés Kindle-re" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "A kért fájl nem olvasható. Esetleg jogosultsági probléma lenne?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "A cím átnevezése \"%(src)s\"-ról \"%(dest)s\"-ra nem sikerült a következő hiba miatt: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "A \"%(file)s\" fájl nem található a Google Drive-on" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "A cím átnevezése \"%(src)s\"-ról \"%(dest)s\"-ra nem sikerült a következő hiba miatt: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "A könyv elérési útja (\"%(path)s\") nem található a Google Drive-on" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Felfedezés" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Be vagy jelentkezve mint: %(nickname)s" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Belépés" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "A token nem található." -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "A token érvényessége lejárt." -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Sikerült! Újra használható az eszköz." -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Legutóbbi könyvek mutatása" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Kelendő könyvek" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Kelendő könyvek mutatása" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Legjobb könyvek" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Legjobbra értékelt könyvek mutatása" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Olvasott könyvek" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Mutassa az olvasva/olvasatlan állapotot" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Olvasatlan könyvek" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Felfedezés" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Mutass könyveket találomra" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Címkék" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Címke választó mutatása" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Sorozatok" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Sorozat választó mutatása" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Szerzők" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Szerző választó mutatása" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Kiadók" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Kiadó választó mutatása" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Nyelvek" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Nyelv választó mutatása" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Sorozat választó mutatása" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Sorozat választó mutatása" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Legutóbbi könyvek mutatása" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Keresés" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Kiadva ezután: " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Kiadva ezelőtt: " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Értékelés <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Értékelés <= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Részletes keresés" @@ -1191,7 +1197,7 @@ msgstr "A könyv el lett távolítva a polcról: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Polc készítése" @@ -1246,45 +1252,45 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Polc: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Hiba a polc megnyitásakor. A polc nem létezik vagy nem elérhető." -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Feladatok" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Várakozás" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Nem sikerült" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Elindítva" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Végrehajtva" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Ismeretlen állapot" @@ -1317,175 +1323,175 @@ msgstr "Új frissítés érhető el. Kattints az alábbi gombra a frissítéshez msgid "No release information available" msgstr "Nincs információ a kiadásról." -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Felfedezés (könyvek találomra)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Kelendő könyvek (legtöbbet letöltöttek)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Kiadó: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Sorozat: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Címke: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Nyelv: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Letöltések" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Először be kell állítani az SMTP levelező beállításokat..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "A könyv sikeresen küldésre lett jelölve a következő címre: %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Hiba történt a könyv küldésekor: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Először be kell állítani a kindle e-mail címet..." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Regisztrálás" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Nem engedélyezett a megadott e-mail cím bejegyzése" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Jóváhagyó levél elküldve az email címedre." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "Be vagy jelentkezve mint: %(nickname)s" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Rossz felhasználó név vagy jelszó!" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Ismeretlen hiba történt. Próbáld újra később!" -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Rossz felhasználó név vagy jelszó!" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Be vagy jelentkezve mint: %(nickname)s" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s profilja" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "A profil frissítve." -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Már létezik felhasználó ehhez az e-mail címhez." @@ -1494,54 +1500,58 @@ msgstr "Már létezik felhasználó ehhez az e-mail címhez." msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Küldés Kindle-re" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Az e-könyv átalakítás nem sikerült: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Az e-könyv átalakítás nem sikerült: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1550,30 +1560,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "Metaadatok szerkesztése" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Feltöltés" @@ -1592,12 +1598,12 @@ msgstr "Felhasználói név" msgid "Email" msgstr "E-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Rendszergazda" @@ -1607,8 +1613,8 @@ msgstr "Rendszergazda" msgid "Password" msgstr "Jelszó" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Letöltés" @@ -1819,13 +1825,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "" @@ -1875,16 +1881,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "csökkentsd" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Több eszerint:" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Nyelv" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Kiadó" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Ismertető:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Előző" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Következő" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Kezdőlap" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Kilépés" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Könyv törlése" @@ -1913,99 +1979,107 @@ msgstr "Konvertálás erre:" msgid "Convert book" msgstr "Könyv konvertálása" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Feltöltés..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Bezárás" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Hiba" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Feltöltés kész, feldolgozás alatt, kérlek várj..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Feltöltés formátuma" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Könyv címe" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Szerző" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Címkék" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Kiadás éve" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Értékelés" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Leírás" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Címkék" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Értékelés" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Borító URL (jpg, borító letöltve és elmentve az adatbázisban, a mező újra üres lesz utána)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Borító feltöltése helyi meghajtóról" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Kiadás éve" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Kiadó" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Nyelv" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Igen" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nem" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Feltöltés formátuma" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Könyv megnézése szerkesztés után" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Metaadatok beszerzése" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2013,38 +2087,32 @@ msgstr "Metaadatok beszerzése" msgid "Save" msgstr "" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Kulcsszó" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Keresési kulcsszó " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Kattints a borítóra a metadatok betöltésére" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Betöltés..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Bezárás" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Forrás" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Keresési hiba!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nincs találat! Próbálj másik kulcsszót." @@ -2153,7 +2221,7 @@ msgid "Enter " msgstr "Regisztrálás" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Biztosan?" @@ -2629,74 +2697,61 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Olvasás böngészőben" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Legyen olvasatlan" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Legyen olvasott" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Legyen olvasatlan" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Olvasva" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Keresés" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Ismertető:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Hozzáadás polchoz" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Metaadatok szerkesztése" @@ -2768,10 +2823,6 @@ msgstr "Tartomány megadása" msgid "Denied Domains (Blacklist)" msgstr "" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Következő" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2792,11 +2843,16 @@ msgstr "" msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Funkciók beállítása" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Vissza a kezdőlapra" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2886,7 +2942,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2895,60 +2951,36 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Kezdőlap" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Navigáció átkapcsolása" -#: cps/templates/layout.html:47 -msgid "Search Library" +#: cps/templates/layout.html:59 +msgid "Simple Theme" msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Felhasználói fiók" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Kilépés" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Feltöltés..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Hiba" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Feltöltés kész, feldolgozás alatt, kérlek várj..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Beállítások" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Böngészés" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Névjegy" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Előző" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Könyv részletei" @@ -3064,7 +3096,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr " " @@ -3073,35 +3105,80 @@ msgstr " " msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web e-könyv katalógus" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Válassz egy felhasználónevet" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Világos" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Sötét" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Vissza" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Szöveg újratördelése amikor az oldalsávok nyitva vannak" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Törlés" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Várakozás" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Függőleges" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Olvasva" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "" @@ -3280,10 +3357,6 @@ msgstr "" msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "" @@ -3300,11 +3373,11 @@ msgstr "Kiadás éve ettől: " msgid "Published Date To" msgstr "Kiadás éve eddig: " -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3341,11 +3414,13 @@ msgstr "Értékelés nagyob mint" msgid "Rating Below" msgstr "Értékelés kisebb mint" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3369,6 +3444,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Nyilvános polc" @@ -3437,15 +3520,19 @@ msgstr "Állapot" msgid "Run Time" msgstr "Futásidő" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3453,6 +3540,10 @@ msgstr "" msgid "Reset user Password" msgstr "Felhasználó jelszavának alaphelyzetbe állítása" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Mutasd a könyveket a következő nyelvvel" diff --git a/cps/translations/id/LC_MESSAGES/messages.mo b/cps/translations/id/LC_MESSAGES/messages.mo index 8af80454..61d06162 100644 Binary files a/cps/translations/id/LC_MESSAGES/messages.mo and b/cps/translations/id/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/id/LC_MESSAGES/messages.po b/cps/translations/id/LC_MESSAGES/messages.po index b425f7c1..ed319d43 100644 --- a/cps/translations/id/LC_MESSAGES/messages.po +++ b/cps/translations/id/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2023-01-21 10:00+0700\n" "Last-Translator: Arief Hidayat\n" "Language: id\n" @@ -16,501 +16,519 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistik" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Server dimulai ulang, harap muat ulang halaman" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Mematikan server, silakan tutup jendela" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Perintah tidak diketahui" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Uji email diantrean untuk dikirim ke %(email), harap periksa Tasks untuk hasilnya" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Tidak diketahui" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Halaman Admin" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Pengaturan Dasar" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Pengaturan Antarmuka" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Kolom Kustom No.%(column)d tidak ada di basis data kaliber" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Edit pengguna" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Semua" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Pengguna tidak ditemukan" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} pengguna berhasil dihapus" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Tampilkan semua" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Permintaan salah" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Nama Tamu tidak dapat diganti" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Tamu tidak dapat memiliki peran ini" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Tidak ada pengguna admin yang tersisa, tidak dapat menghapus peran admin" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Nilai harus benar atau salah" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Peran tidak valid" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr " Tamu tidak dapat mengakses tampilan ini" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr " Tampilan tidak valid" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Lokal Tamu ditentukan secara otomatis dan tidak dapat disetel" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Tidak Ada Lokal yang Valid Diberikan" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Tidak Ada Bahasa Buku yang Valid Diberikan" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parameter tidak ditemukan" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Kolom Baca Tidak Valid" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Kolom Dibatasi Tidak Valid" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Pengaturan Calibre-Web telah diperbarui" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Apakah Anda yakin ingin menghapus Token Kobo?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Apakah Anda yakin ingin menghapus domain ini?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Apakah Anda yakin ingin menghapus pengguna ini?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Apakah Anda yakin ingin menghapus rak ini?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Apakah Anda yakin ingin merubah lokalisasi untuk pengguna yang dipilih?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Apakah Anda yakin ingin merubah bahasa buku yang terlihat untuk pengguna yang dipilih?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Apakah Anda yakin ingin merubah peran untuk pengguna yang dipilih?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Apakah Anda yakin ingin mengubah batasan yang dipilih untuk pengguna yang dipilih?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Apakah Anda yakin ingin merubah batasan visibilitas untuk pengguna yang dipilih?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Apakah Anda yakin ingin mengubah perilaku sinkronisasi rak untuk pengguna yang dipilih?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Apakah Anda yakin ingin mengubah lokasi perpustakaan Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calibre-Web akan mencari Sampul yang diperbarui dan memperbarui Thumbnail Sampul, ini mungkin memakan waktu cukup lama?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Apakah Anda yakin ingin menghapus database sinkronisasi Calibre-Web untuk memaksakan sinkronisasi penuh dengan Kobo Reader Anda?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Tolak" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Izinkan" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} entri sinkronisasi dihapus" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Tag tidak ditemukan" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Tindakan Tidak Valid" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json Tidak Diatur Untuk Aplikasi Web" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Lokasi Logfile tidak Valid, Harap Masukkan Jalur yang Benar" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Akses Logfile Catatan tidak Valid, Harap Masukkan Jalur yang Benar" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Harap Masukkan Provider LDAP, Port, DN dan User Obect Identifier" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Masukkan Akun Layanan LDAP dan Kata Sandi" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Masukkan Akun Layanan LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Filter Objek Grup LDAP Harus Memiliki Satu Pengidentifikasi Format \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Filter Objek Grup LDAP Memiliki Tanda kurung yang Tak Berpasangan" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Filter Objek Pengguna LDAP harus Memiliki Satu Pengidentifikasi Format \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Filter Objek Pengguna LDAP Memiliki Tanda kurung yang Tak Berpasangan" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Filter Pengguna Anggota LDAP harus Memiliki Satu Pengenal Format \"%s\"" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Filter Pengguna Anggota LDAP Memiliki Tanda Kurung yang Tak Berpasangan" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "Lokasi LDAP Sertifikat CA, Sertifikat, atau Kunci tidak Valid, Harap Masukkan Jalur yang Benar " -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Tambah Pengguna Baru" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Edit Pengaturan Server Email" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Kesalahan basis data: %(error)s" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Uji email diantrean untuk dikirim ke %(email), harap periksa Tasks untuk hasilnya" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Terjadi kesalahan saat mengirim email tes: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Harap atur alamat email Anda terlebih dahulu.." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Setelan server email diperbarui" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Edit Pengaturan Tugas Terjadwal" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Waktu mulai tidak valid untuk tugas yang ditentukan" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Durasi tidak valid untuk tugas yang ditentukan" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Pengaturan tugas terjadwal diperbarui" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Terjadi kesalahan yang tidak diketahui. Coba lagi nanti." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Pengaturan DB tidak dapat ditulisi" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Edit pengguna %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Kata sandi untuk pengaturan ulang pengguna %(user) " -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Harap atur pengaturan email SMTP terlebih dahulu..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Penampil berkas log" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Meminta paket pembaruan" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Mengunduh paket pembaruan" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Mengekstrak paket pembaruan" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Mengganti berkas" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Koneksi basis data ditutup" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Menghentikan server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Pembaruan selesai, silakan tekan OK dan muat ulang halaman" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Pembaruan gagal:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Kesalahan HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Kesalahan koneksi" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Batas waktu saat membuat koneksi" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Kesalahan umum" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "Berkas pembaruan tidak dapat disimpan di direktori temp" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Berkas tidak dapat diganti selama pembaruan" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Gagal mengekstrak setidaknya Satu Pengguna LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Gagal Membuat Sedikitnya Satu Pengguna LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Kesalahan: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Error: Tidak ada pengguna yang dikembalikan sebagai respons dari server LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Setidaknya Satu Pengguna LDAP Tidak Ditemukan di Basis Data" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Pengguna Berhasil Diimpor" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Lokasi Basis Data tidak Valid, Harap Masukkan Jalur yang Benar" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Basis Data tidak dapat ditulisi" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Lokasi keyfile tidak Valid, Harap Masukkan Jalur yang Benar " -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Lokasi Sertifikat tidak Valid, Harap Masukkan Jalur yang Benar " -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Pengaturan Basis Data diperbarui" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Pengaturan Basis Data" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Harap masukkan seluruh isian!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Email bukan dari domain yang valid" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Tambahkan pengguna baru" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Pengguna '%(user)s' telah dibuat" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Ditemukan akun yang ada untuk alamat email atau nama ini." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Pengguna '%(nick)s' telah dihapus" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Tidak dapat menghapus Pengguna Tamu" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Tidak ada pengguna admin tersisa, tidak dapat menghapus pengguna" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "Alamat email tidak boleh kosong dan harus berupa email yang valid" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Pengguna '%(nick)s' diperbarui" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Telusuri" + #: cps/converter.py:31 msgid "not installed" msgstr "belum dipasang" @@ -519,128 +537,123 @@ msgstr "belum dipasang" msgid "Execution permissions missing" msgstr "Izin eksekusi hilang" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Kolom Kustom No.%(column)d tidak ada di basis data kaliber" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Tidak ada" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Ups! Judul buku yang dipilih tidak tersedia. Berkas tidak ada atau tidak dapat diakses" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Pengguna tidak berhak mengganti sampul" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "IDは大文字小文字を区別しません。元のIDを上書きします" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadata berhasil diperbarui" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Kesalahan pengeditan buku: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Berkas %(file)s telah diunggah" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Format sumber atau tujuan untuk konversi tidak ada" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Buku berhasil diantrekan untuk dikonversi ke %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Terjadi kesalahan saat mengonversi buku ini: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Buku yang diunggah mungkin ada di perpustakaan, pertimbangkan untuk mengubahnya sebelum mengunggah yang baru: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Ups! Judul buku yang dipilih tidak tersedia. Berkas tidak ada atau tidak dapat diakses" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Pengguna tidak berhak mengganti sampul" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "IDは大文字小文字を区別しません。元のIDを上書きします" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "'%(langname)s' bukan bahasa yang valid" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadata berhasil diperbarui" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Kesalahan pengeditan buku: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Buku yang diunggah mungkin ada di perpustakaan, pertimbangkan untuk mengubahnya sebelum mengunggah yang baru: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Ekstensi berkas '%(ext)s' tidak diizinkan untuk diunggah ke server ini" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Ekstensi berkas '%(ext)s' tidak diizinkan untuk diunggah ke server ini" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Berkas yang akan diunggah harus memiliki ekstensi" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Berkas %(filename)s tidak dapat disimpan ke direktori temp" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Gagal Memindahkan Berkas Sampul %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Format Buku Berhasil Dihapus" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Buku Berhasil Dihapus" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Anda tidak memiliki izin untuk menghapus buku" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "edit metadata" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s dilewati karena bukan angka yang valid" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Pengguna tidak memiliki izin untuk mengunggah format berkas tambahan" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Gagal membuat jalur %(path)s (Izin ditolak)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Gagal menyimpan berkas %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Format berkas %(ext)s ditambahkan ke %(book)s" @@ -653,485 +666,478 @@ msgstr "Pengaturan Google Drive belum selesai, coba nonaktifkan dan aktifkan kem msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Domain panggilan balik tidak diverifikasi, ikuti langkah-langkah untuk memverifikasi domain di konsol pengembang google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s format tidak ditemukan untuk id buku: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s tidak ditemukan di Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s tidak ditemukan: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "Kirim ke E-Reader" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Email ini telah dikirim melalui Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Email tes Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Email tes" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Memulai dengan Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Email pendaftaran untuk pengguna: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Ubah %(orig)s menjadi %(format)s dan kirim ke E-Reader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Kirim %(format)s ke E-Reader" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%%(buku)s telah dikirim ke E-Reader" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Berkas yang diminta tidak dapat dibaca. Mungkin izinnya salah?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Status baca tidak bisa disetel: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Menghapus folder buku untuk buku %(id)s gagal, jalur memiliki subfolder: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Gagal menghapus buku %(id)s: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Menghapus buku %(id)s hanya dari basis data, jalur buku di basis data tidak valid: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Ganti nama pengarang dari: '%(src)s' menjadi '%(dest)s' gagal dengan kesalahan: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Berkas %(file)s tidak ditemukan di Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Ganti nama judul dari: '%(src)s' menjadi '%(dest)s' gagal dengan kesalahan: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Jalur buku %(path)s tidak ditemukan di Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Nama pengguna ini sudah digunakan" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Format alamat email tidak valid" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "Modul 'advocate' Python tidak diinstal tetapi diperlukan untuk unggahan sampul" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Kesalahan Mengunduh Sampul" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Kesalahan Format Sampul" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Anda tidak diizinkan mengakses localhost atau jaringan lokal untuk unggahan sampul" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Gagal membuat jalur untuk sampul" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Berkas sampul bukan berkas gambar yang valid, atau tidak dapat disimpan" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Hanya berkas jpg/jpeg/png/webp/bmp yang didukung sebagai berkas sampul" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Konten berkas sampul tidak valid" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Hanya berkas jpg/jpeg yang didukung sebagai berkas sampul" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Sampul" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Berkas biner unrar tidak ditemukan" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Kesalahan saat menjalankan UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "Basis Data tidak dapat ditulisi" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Izin eksekusi hilang" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Kesalahan saat menjalankan UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Antrian semua buku untuk cadangan metadata" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Mohon akses calibre-web dari non localhost untuk mendapatkan api_endpoint yang valid untuk perangkat kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Pengaturan Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Daftar dengan %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Anda sekarang login sebagai: %(nickname)s" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Tautan ke %(oauth)s Berhasil" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Login gagal, Tidak Ada Pengguna yang Tertaut Dengan Akun OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Membatalkan tautan ke %(oauth)s Berhasil" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Membatalkan tautan ke %(oauth)s Gagal" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Tidak Tertaut ke %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Gagal masuk dengan GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Gagal mengambil info pengguna dari GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Gagal masuk dengan Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Gagal mengambil info pengguna dari Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Kesalahan GitHub Oauth, silakan coba lagi nanti." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Kesalahan GitHub OAuth: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Kesalahan Google Oauth, harap coba lagi nanti." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Kesalahan Google OAuth: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{}★" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Masuk" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token tidak ditemukan" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token telah kedaluwarsa" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Berhasil! Silakan kembali ke perangkat Anda" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Buku" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Tampilkan buku terbaru" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Buku Populer" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Tampilkan Buku Populer" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Buku yang Diunduh" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Tampilkan Buku yang Diunduh" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Buku Berperingkat Teratas" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Tampilkan Buku Berperingkat Teratas" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Buku Telah Dibaca" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Tampilkan sudah dibaca dan belum dibaca" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Buku yang Belum Dibaca" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Tampilkan belum dibaca" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Temukan" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Tampilkan Buku Acak" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategori" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Tampilkan pilihan kategori" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Seri" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Tampilkan pilihan seri" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Penulis" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Tampilkan pilihan penulis" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Penerbit" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Tampilkan pilihan penerbit" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Bahasa" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Tampilkan pilihan bahasa" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Peringkat" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Tampilkan pilihan peringkat" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Format berkas" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Tampilkan pilihan format berkas" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Buku yang Diarsipkan" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Tampilkan buku yang diarsipkan" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Daftar Buku" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Tampilkan Daftar Buku" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Telusuri" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Terbit setelah " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Terbit sebelum " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Peringkat ≤ %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Peringkat ≥ %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Status Baca = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Terjadi kesalahan saat mencari kolom khusus, harap mulai ulang Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Penelusuran Lanjutan" @@ -1186,7 +1192,7 @@ msgstr "Buku telah dihapus dari rak: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Maaf Anda tidak diizinkan untuk menghapus buku dari rak ini: %(sname)s" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Buat Rak" @@ -1239,45 +1245,45 @@ msgstr "Rak publik dengan nama '%(title)s' sudah ada." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Rak pribadi dengan nama '%(title)s' sudah ada." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Rak: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Terjadi kesalahan saat membuka rak. Rak tidak ada atau tidak dapat diakses" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tugas" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Menunggu" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Gagal" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Dimulai" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Selesai" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Berakhir" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Dibatalkan" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Status Tidak Diketahui" @@ -1310,178 +1316,178 @@ msgstr "Pembaruan tersedia. Klik tombol di bawah untuk memperbarui ke versi: %(v msgid "No release information available" msgstr "Tidak ada informasi rilis yang tersedia" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Temukan (Buku Acak)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Buku Populer (Paling Banyak Diunduh)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Buku telah diunduh oleh %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Penulis: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Penerbit: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Seri: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Peringkat: Tidak ada" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Peringkat: %(rating)s★" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Format berkas: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategori: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Bahasa: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Unduhan" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Daftar peringkat" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Daftar format berkas" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Harap atur pengaturan email SMTP terlebih dahulu..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Buku telah diantrikan untuk dikirim ke %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Oops! Terjadi kesalahan saat mengirim buku: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Harap perbarui profil Anda dengan alamat e-mail Kirim ke Kindle yang valid." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Daftar" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Server email belum diatur, silakan hubungi administrator!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Server email belum diatur, silakan hubungi administrator!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Alamat email Anda tidak diizinkan untuk mendaftar" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "E-mail konfirmasi telah dikirimkan ke alamat email Anda." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Tidak dapat mengaktifkan autentikasi LDAP." -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "Anda sekarang login sebagai: %(nickname)s" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Login Pengganti sebagai: '%(nickname)s', Server LDAP tidak dapat dijangkau, atau pengguna tidak diketahui." -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Tidak dapat login: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Pengguna atau Kata Sandi salah" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Kata Sandi baru telah dikirimkan ke alamat email Anda" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Terjadi kesalahan yang tidak diketahui. Coba lagi nanti." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Harap masukkan pengguna valid untuk mengatur ulang kata sandi" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Anda sekarang login sebagai: %(nickname)s" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Profil %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profil diperbarui" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Ditemukan akun yang ada untuk alamat email ini" @@ -1489,54 +1495,58 @@ msgstr "Ditemukan akun yang ada untuk alamat email ini" msgid "Found no valid gmail.json file with OAuth information" msgstr "Tidak ditemukan berkas gmail.json yang valid dengan informasi OAuth" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%%(buku)s telah dikirim ke E-Reader" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre ebook-convert %(tool)s tidak ditemukan" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s format tidak ditemukan dalam disk" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Konverter ebook gagal dengan kesalahan yang tidak diketahui." -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kebupify-converter gagal: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Berkas yang telah dikonversi tidak ditemukan atau terdapat duplikat dalam folder %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ebook-converter gagal: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre gagal dengan kesalahan: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ebook-converter gagal: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Konversi" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Menghubungkan kembali basis data Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "Email" @@ -1545,30 +1555,26 @@ msgstr "Email" msgid "Backing up Metadata" msgstr "Mencadangkan Metadata" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "%(count)s thumbnail sampul dibuat" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Thumbnail Sampul" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "{0} thumbnail seri dihasilkan" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Menghapus cache thumbnail sampul" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Unggah" @@ -1587,11 +1593,11 @@ msgstr "Nama Pengguna" msgid "Email" msgstr "Alamat Email" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "Alamat E-mail untuk Kirim ke E-Reader" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Admin" @@ -1601,8 +1607,8 @@ msgstr "Admin" msgid "Password" msgstr "Kata Sandi" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Unduh" @@ -1813,13 +1819,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Batal" @@ -1869,16 +1875,76 @@ msgstr "Urutkan menurut tanggal penerbitan, yang terbaru dulu" msgid "Sort according to publishing date, oldest first" msgstr "Urutkan menurut tanggal penerbitan, yang terlama dulu" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "kurangi" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Lainnya oleh" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Buku" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Bahasa" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Penerbit" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Diterbitkan" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Deskripsi:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Sebelumnya" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Selanjutnya" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Tidak ada hasil yang ditemukan" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Beranda" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Cari di Pustaka" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Keluar" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Hapus" @@ -1907,99 +1973,107 @@ msgstr "Konversi ke:" msgid "Convert book" msgstr "Konversi buku" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Mengunggah..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Tutup" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Terjadi Kesalahan" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Unggahan selesai, harap tunggu, data sedang diproses..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Format Unggahan" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Judul Buku" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Penulis" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Tag" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID Seri" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Tanggal Terbit" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Rating" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Deskripsi" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Pengidentifikasi" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Tipe Pengidentifikasi" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Nilai Pengidentifikasi" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Hapus" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Tambah Pengidentifikasi" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Tag" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID Seri" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Rating" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Ambil Sampul dari URL (JPEG - Gambar akan diunduh dan disimpan dalam basis data)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Unggah Sampul dari disk lokal" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Tanggal Terbit" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Penerbit" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Bahasa" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ya" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Tidak" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Format Unggahan" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Tampilkan Buku setelah Disimpan" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Ambil Metadata" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2007,37 +2081,31 @@ msgstr "Ambil Metadata" msgid "Save" msgstr "Simpan" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Kata Kunci" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Cari Kata kunci" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Klik sampul untuk memuat metadata ke formulir" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Memuat..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Tutup" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Sumber" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Kesalahan pencarian!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Tidak ada hasil yang ditemukan! Silakan coba kata kunci lain." @@ -2144,7 +2212,7 @@ msgid "Enter " msgstr "入力: " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Apakah Anda yakin?" @@ -2620,74 +2688,61 @@ msgstr "Tambahkan Tag yang Diizinkan/Ditolak" msgid "Add Allowed/Denied custom column values" msgstr "Tambahkan nilai kolom khusus yang Diizinkan/Ditolak" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Baca di Peramban" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Dengarkan di Browser" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Buku" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Diterbitkan" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Tandai sebagai Belum dibaca" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Tandai sebagai dibaca" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Tandai sebagai Belum dibaca" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Baca" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Pulihkan dari arsip" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Tambahkan ke Arsip" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Diarsipkan" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Deskripsi:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Tambah ke rak" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Publik)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Edit Metadata" @@ -2760,10 +2815,6 @@ msgstr "Masukkan Nama Domain" msgid "Denied Domains (Blacklist)" msgstr "Domain yang Ditolak (Daftar Hitam)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Selanjutnya" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Buka berkas .kobo/Kobo/Kobo eReader.conf di editor teks dan tambahkan (atau edit):" @@ -2784,11 +2835,16 @@ msgstr "Instans Calibre-Web belum diatur, harap hubungi administrator Anda" msgid "Create Issue" msgstr "Buat Isu" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Pengaturan Basis Data" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Kembali ke Beranda" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Keluar Pengguna" @@ -2878,7 +2934,7 @@ msgstr "Buku yang diurutkan menurut Peringkat" msgid "Books ordered by file formats" msgstr "Buku yang diurutkan menurut format berkas" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Rak" @@ -2887,60 +2943,37 @@ msgstr "Rak" msgid "Books organized in shelves" msgstr "本棚に整理された本" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Beranda" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Alihkan Navigasi" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Cari di Pustaka" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Sederhana" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Akun" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Keluar" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Mengunggah..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Terjadi Kesalahan" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Unggahan selesai, harap tunggu, data sedang diproses..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Pengaturan" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Harap jangan segarkan halaman" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Jelajahi" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Tentang" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Sebelumnya" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Detail Buku" @@ -3056,7 +3089,7 @@ msgstr "Direktori Induk" msgid "Select" msgstr "Pilih" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3064,34 +3097,80 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Katalog eBook Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "Pembaca EPUB" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Pilih nama pengguna" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Terang" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Gelap" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sepia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Hitam" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Reflow teks saat sidebar terbuka." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Hapus" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Menunggu" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Vertical" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Baca" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Kolom Baca Tidak Valid" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Pembaca Komik" @@ -3269,10 +3348,6 @@ msgstr "Tautan verifikasi ini akan kedaluwarsa dalam 10 menit." msgid "Generate Series Cover Thumbnails" msgstr "Hasilkan Thumbnail Sampul Seri" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Tidak ada hasil yang ditemukan" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Istilah Penelusuran:" @@ -3289,11 +3364,11 @@ msgstr "Tanggal Diterbitkan Dari" msgid "Published Date To" msgstr "Tanggal Diterbitkan Hingga" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3329,11 +3404,13 @@ msgstr "Peringkat Diatas" msgid "Rating Below" msgstr "Peringkat Dibawah" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Dari:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Hingga:" @@ -3357,6 +3434,16 @@ msgstr "Nonaktifkan Ubah urutan" msgid "Enable Change order" msgstr "Aktifkan Ubah urutan" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Urutkan menurut tanggal buku, terbaru dulu" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Urutkan menurut tanggal buku, terlama dulu" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Bagikan dengan Semua Orang" @@ -3425,15 +3512,20 @@ msgstr "Kemajuan" msgid "Run Time" msgstr "Waktu Jalan" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Gabungkan" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Tindakan" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Tugas ini akan dibatalkan. Setiap kemajuan yang dibuat oleh tugas ini akan disimpan." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Jika ini adalah tugas terjadwal, ini akan dijalankan ulang selama waktu terjadwal berikutnya." @@ -3441,6 +3533,10 @@ msgstr "Jika ini adalah tugas terjadwal, ini akan dijalankan ulang selama waktu msgid "Reset user Password" msgstr "Atur ulang kata sandi pengguna" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Bahasa Buku" diff --git a/cps/translations/it/LC_MESSAGES/messages.mo b/cps/translations/it/LC_MESSAGES/messages.mo index 9046e1c9..aceb45c1 100644 Binary files a/cps/translations/it/LC_MESSAGES/messages.mo and b/cps/translations/it/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/it/LC_MESSAGES/messages.po b/cps/translations/it/LC_MESSAGES/messages.po index 7a0ae3ea..57889fc6 100644 --- a/cps/translations/it/LC_MESSAGES/messages.po +++ b/cps/translations/it/LC_MESSAGES/messages.po @@ -7,8 +7,8 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" -"PO-Revision-Date: 2024-07-05 04:34+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-12-15 06:37+0100\n" "Last-Translator: Massimo Pissarello \n" "Language: it\n" "Language-Team: Italian <>\n" @@ -16,497 +16,515 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistiche" -#: cps/admin.py:150 +#: cps/admin.py:151 msgid "Server restarted, please reload page." msgstr "Server riavviato, ricarica la pagina." -#: cps/admin.py:152 +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." msgstr "Esecuzione dell'arresto del server, chiudi la finestra." -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "Tutto OK! Database riconnesso" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Comando sconosciuto" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Tutto OK! Libri in coda per il backup dei metadati, controlla le attività per il risultato" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Sconosciuto" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Pagina di amministrazione" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Configurazione di base" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Configurazione dell'interfaccia utente" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "La colonna personalizzata no.%(column)d non esiste nel database di Calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Modifica utenti" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Tutti" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Utente non trovato" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "utenti eliminati correttamente" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Mostra tutto" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Richiesta non valida" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Il nome dell'utente Guest (ospite) non può essere modificato" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "L'utente Guest (ospite) non può avere questo ruolo" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Non rimarrebbe nessun utente amministratore, non è possibile rimuovere il ruolo di amministratore" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Il valore deve essere vero o falso" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Ruolo non valido" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "L'utente Guest (ospite) non può visualizzare questa schermata" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Visualizzazione non valida" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Le impostazioni locali dell'utente Guest (ospite) sono determinate automaticamente e non possono essere configurate" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Nessuna lingua valida specificata" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Nessun libro valido per la lingua specificata" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parametro non trovato" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Colonna di lettura non valida" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Colonna con restrizioni non valida" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "La configurazione di Calibre-Web è stata aggiornata" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Vuoi veramente eliminare il token di Kobo?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Vuoi veramente eliminare questo dominio?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Vuoi veramente eliminare questo utente?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Sei sicuro di voler eliminare questo scaffale?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Sei sicuro di voler cambiare le impostazioni internazionali degli utenti selezionati?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Sei sicuro di voler cambiare le lingue visibili del libro per gli utenti selezionati?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Sei sicuro di voler cambiare il ruolo selezionato per gli utenti selezionati?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Sei sicuro di voler cambiare le restrizioni selezionate per gli utenti selezionati?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Sei sicuro di voler cambiare le restrizioni di visibilità selezionate per gli utenti selezionati?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Sei sicuro di voler cambiare il comportamento di sincronizzazione dello scaffale per gli utenti selezionati?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Sei sicuro di voler cambiare la posizione della biblioteca di Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calibre-Web cercherà le copertine aggiornate e aggiornerà le miniature delle copertine, ma ci vorrà un po' di tempo." -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Sei sicuro di voler eliminare il database sincronizzato di Calibre-Web e forzare una sincronizzazione completa con il tuo lettore Kobo?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Nega" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Consenti" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} voci di sincronizzazione eliminate" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Etichetta non trovata" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Azione non valida" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json non è configurato per Web Application" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "La posizione del file di log non è valida, per favore indica il percorso corretto" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "La posizione del file del log di accesso non è valida, indica il percorso corretto" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Inserisci un provider LDAP, una porta, un DN e un identificatore oggetto utente" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Inserisci un account e una password del servizio LDAP" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Inserisci un account di servizio LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Il filtro oggetto gruppo LDAP deve avere un identificatore di formato \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Il filtro oggetto gruppo LDAP ha parentesi senza corrispondenza" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Il filtro oggetto utente LDAP deve avere un identificatore di formato \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Il filtro oggetto utente LDAP ha parentesi senza corrispondenza" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Il filtro utente membro LDAP deve avere un identificatore di formato \"%s\"" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Il filtro utente membro LDAP ha parentesi senza corrispondenza" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "Il certificato CA LDAP, il certificato o la posizione della chiave non sono validi. Inserisci il percorso corretto" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Aggiungi nuovo utente" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Modifica le impostazioni del server e-mail" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "Tutto OK! Account Gmail verificato." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Errore nel database: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "L'e-mail di prova è stato accodata correttamente per essere spedita a %(email)s, controlla il risultato in Attività" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Si è verificato un errore nell'invio dell'e-mail di prova: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Configura prima il tuo indirizzo e-mail..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Le impostazioni del server e-mail sono state aggiornate" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Modifica le impostazioni delle attività pianificate" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Ora di inizio non valida per l'attività specificata" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Durata non valida per l'attività specificata" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Impostazioni delle attività pianificate aggiornate" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Si è verificato un errore sconosciuto. Per favore riprova più tardi." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Il DB delle impostazioni non è scrivibile" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Modifica utente %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "Tutto OK! Password reimpostata per l'utente %(user)s" -#: cps/admin.py:1451 +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." msgstr "Per favore configura le impostazioni della posta SMTP." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Visualizzatore del file di log" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Richiesta del pacchetto di aggiornamento" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Download del pacchetto di aggiornamento" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Decompressione del pacchetto di aggiornamento" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Sostituzione dei file" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Le connessioni al database sono chiuse" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Arresto del server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Aggiornamento terminato, premi OK e ricarica la pagina" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Aggiornamento non riuscito:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Errore HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Errore di connessione" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Tempo scaduto nello stabilire la connessione" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Errore generale" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "Il file di aggiornamento non può essere salvato nella cartella temporanea" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Impossibile sostituire i file durante l'aggiornamento" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Impossibile estrarre almeno un utente LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Impossibile creare almeno un utente LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Errore: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Errore: nessun utente restituito in risposta dal server LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Almeno un utente LDAP non è stato trovato nel database" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} utente importato correttamente" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "Percorso dei libri non valido" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "La posizione del DB non è valida, per favore indica il percorso corretto" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Il DB non è scrivibile" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "La posizione del Keyfile non è valida. Inserisci il percorso corretto" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "La posizione del Certfile non è valida, indica il percorso corretto" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "La lunghezza della password deve essere compresa tra 1 e 40" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Impostazioni del database aggiornate" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Configurazione del database" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Per favore completa tutti i campi." -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "L'e-mail non proviene da un dominio valido" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Aggiungi nuovo utente" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "L'utente '%(user)s' è stato creato" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Esiste già un account per questa e-mail o nome." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "L'utente '%(nick)s' è stato eliminato" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Impossibile eliminare l'utente Guest (ospite)" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Non rimarrebbe nessun utente amministratore, non è possibile eliminare l'utente" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "L'e-mail non può essere vuota e deve essere un'e-mail valida" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "L'utente '%(nick)s' è stato aggiornato" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Cerca" + #: cps/converter.py:31 msgid "not installed" msgstr "non installato" @@ -515,127 +533,122 @@ msgstr "non installato" msgid "Execution permissions missing" msgstr "Mancano i permessi di esecuzione" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "La colonna personalizzata no.%(column)d non esiste nel database di Calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Nessuna" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Il libro selezionato non è disponibile. Il file non esiste o non è accessibile" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "L'utente non ha i permessi per caricare le copertine" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Gli identificatori non fanno distinzione tra maiuscole e minuscole e sovrascrivono il vecchio identificatore" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadati aggiornati correttamente" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Errore nella modifica del libro: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Il file %(file)s è stato caricato" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Manca il formato di origine o di destinazione per la conversione" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Libro accodato correttamente per essere convertito in %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Si è verificato un errore durante la conversione del libro: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Probabilmente il libro caricato esiste già nella biblioteca, cambialo prima di caricarlo di nuovo:" +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Il libro selezionato non è disponibile. Il file non esiste o non è accessibile" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "L'utente non ha i permessi per caricare le copertine" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Gli identificatori non fanno distinzione tra maiuscole e minuscole e sovrascrivono il vecchio identificatore" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s non è una lingua valida" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadati aggiornati correttamente" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Errore nella modifica del libro: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Probabilmente il libro caricato esiste già nella biblioteca, cambialo prima di caricarlo di nuovo:" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" msgstr "Non è consentito caricare questo tipo di file su questo server" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Non è consentito caricare l'estensione del file '%(ext)s' su questo server" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Il file da caricare deve avere un'estensione" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Il file %(filename)s non può essere salvato nella cartella temporanea" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Impossibile spostare il file della copertina %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Il formato del libro è stato eliminato correttamente" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Il libro è stato eliminato correttamente" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Ti mancano le autorizzazioni per eliminare i libri" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "modifica i metadati" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" -msgstr "%(seriesindex)s non è un numero valido, lo salto" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "Indice serie: %(seriesindex)s non è un numero valido, lo salto" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "L'utente non ha i permessi per caricare formati di file aggiuntivi" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Impossibile creare il percorso %(path)s (autorizzazione negata)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Impossibile archiviare il file %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Formato file %(ext)s aggiunto a %(book)s" @@ -648,468 +661,461 @@ msgstr "Configurazione di Google Drive non completata, prova a disattivare e att msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Il dominio di callback non è stato verificato, segui i passaggi per verificare il dominio nella console per sviluppatori di Google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "Formato %(format)s non trovato per l'ID libro: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s non trovato su Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s non trovato: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "Invia all'eReader" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "Questa e-mail è stata inviata tramite Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "E-mail di prova di Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "E-mail di prova" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Inizia con Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "E-mail di registrazione per l'utente: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Converti %(orig)s in %(format)s e invia all'eReader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, python-format msgid "Send %(format)s to eReader" msgstr "Invia %(format)s all'eReader" -#: cps/helper.py:227 +#: cps/helper.py:230 #, python-format msgid "%(book)s send to eReader" msgstr "%(book)s inviato all'eReader" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Il file richiesto non può essere letto. I permessi sono corretti?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Impossibile impostare lo stato di lettura: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Eliminazione della cartella di libri per il libro %(id)s non riuscita, il percorso ha sottocartelle: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Eliminazione del libro %(id)s non riuscita: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Eliminazione del libro %(id)s solo dal database, percorso del libro nel database non valido: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "La modifica dell'autore da '%(src)s' a '%(dest)s' è terminata con l'errore: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Il file %(file) non è stato trovato su Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "La modifica del titolo da '%(src)s' a '%(dest)s' è terminata con l'errore: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Il percorso del libro %(path)s non è stato trovato su Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "Trovato un account esistente per questo indirizzo e-mail" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Questo nome utente è già utilizzato" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "Formato dell'indirizzo e-mail non valido" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "La password non è conforme alle regole di convalida della password" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "Il modulo Python \"advocate\" non è installato ma è necessario per il caricamento delle copertine" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Errore nello scaricare la copertina" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Errore nel formato della copertina" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Non ti è consentito accedere all'host locale o alla rete locale per caricare le copertine" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Impossibile creare il percorso per la copertina" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Il file della copertina non è in un formato di immagine valido o non può essere salvato" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Solo i file jpg/jpeg/png/webp/bmp sono supportati come file di copertina" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Contenuto del file di copertina non valido" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Solo i file jpg/jpeg sono supportati come file di copertina" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" msgstr "Copertina" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "File binario UnRar non trovato" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "Errore nell'eseguire UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "Impossibile trovare la cartella specificata" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "Specifica una cartella, non un file" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "Eseguibili di Calibre non validi" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "File eseguibili di Calibre mancanti: %(missing)s" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Permessi di esecuzione mancanti: %(missing)s" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "Errore durante l'esecuzione di Calibre" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Metti in coda tutti i libri per il backup dei metadati" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Accedi a Calibre-Web da un host non locale per ottenere un api_endpoint valido per il dispositivo Kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Configurazione di Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Registra con %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Tutto OK! Ora sei connesso come: %(nickname)s" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Collegamento riuscito a %(oauth)s" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Accesso non riuscito, non c'è nessun utente collegato all'account OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Scollegamento da %(oauth)s riuscito" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Scollegamento da %(oauth)s non riuscito" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Non collegato a %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Impossibile accedere con GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Impossibile recuperare le informazioni dell'utente da GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Impossibile accedere con Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Impossibile recuperare le informazioni dell'utente da Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Errore GitHub Oauth, riprova più tardi." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Errore GitHub Oauth: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Errore OAuth di Google, riprova più tardi." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Errore OAuth di Google: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} stelle" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Accesso" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token non trovato" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Il token è scaduto" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Tutto OK! Torna al tuo dispositivo" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Libri" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Mostra Libri recenti" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Libri hot" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Mostra Libri hot" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Libri scaricati" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Mostra Libri scaricati" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Libri più votati" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Mostra Libri più votati" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Libri letti" -#: cps/render_template.py:64 +#: cps/render_template.py:63 msgid "Show Read and Unread" msgstr "Mostra Libri letti e Libri da leggere" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Libri da leggere" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Mostra da leggere" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Libri casuali" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Mostra Libri casuali" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Categorie" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" msgstr "Mostra sezione Categorie" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Serie" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" msgstr "Mostra sezione Serie" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autori" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" msgstr "Mostra sezione Autori" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Editori" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" msgstr "Mostra sezione Editori" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Lingue" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" msgstr "Mostra sezione Lingue" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Valutazioni" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" msgstr "Mostra sezione Valutazioni" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formati di file" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" msgstr "Mostra sezione Formati di file" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Libri archiviati" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "Mostra Libri archiviati" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Elenco libri" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Mostra Elenco libri" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Cerca" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Pubblicato dopo il " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Pubblicato prima del " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Valutazione <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Valutazione >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "Stato di lettura = '%(status)s'" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Errore nella ricerca delle colonne personalizzate. Per favore riavvia Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Ricerca avanzata" @@ -1164,7 +1170,7 @@ msgstr "Il libro è stato rimosso dallo scaffale: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Spiacente, ma non sei autorizzato a rimuovere libri da questo scaffale" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Crea uno scaffale" @@ -1217,45 +1223,45 @@ msgstr "Esiste già uno scaffale pubblico con il nome '%(title)s'." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Esiste già uno scaffale privato con il nome '%(title)s'." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Scaffale: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Errore nell'apertura dello scaffale. Lo scaffale non esiste o non è accessibile" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Attività" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Attendi" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Non riuscito" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Avviato" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Finito" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Terminato" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Annullato" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Stato sconosciuto" @@ -1288,169 +1294,169 @@ msgstr "È disponibile un nuovo aggiornamento. Fai clic sul pulsante in basso pe msgid "No release information available" msgstr "Nessuna informazione disponibile sulla versione" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Libri casuali" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Libri hot (i più scaricati)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Libri scaricati da %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Autore: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Editore: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Serie: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Valutazione: nessuna" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Valutazione: %(rating)s stelle" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Formato file: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Categoria: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Lingua: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Scaricati" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Elenco valutazioni" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Elenco formati di file" -#: cps/web.py:1249 +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." msgstr "Configura prima le impostazioni della posta SMTP..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Tutto OK! Libro in coda per l'invio a %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Si è verificato un errore durante l'invio del libro: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Per favore aggiorna il tuo profilo con un'e-mail eReader valida." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "Attendi un minuto per registrare il prossimo utente" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registrati" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "Il server e-mail non è configurato, per favore contatta l'amministratore" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Il server e-mail non è configurato, per favore contatta l'amministratore" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "La tua email non è consentita." -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Tutto OK! L'e-mail di conferma è stata inviata." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "Impossibile attivare l'autenticazione LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "Attendi un minuto prima del prossimo accesso" -#: cps/web.py:1400 +#: cps/web.py:1408 #, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "ora sei connesso come: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Accesso di riserva come: '%(nickname)s', il server LDAP non è raggiungibile o l'utente è sconosciuto" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "Impossibile accedere: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" msgstr "Nome utente o password errati" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "La nuova password è stata inviata al tuo indirizzo e-mail" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "Si è verificato un errore sconosciuto, riprova più tardi." -#: cps/web.py:1429 +#: cps/web.py:1437 msgid "Please enter valid username to reset password" msgstr "Inserisci un nome utente valido per reimpostare la password" -#: cps/web.py:1437 +#: cps/web.py:1445 #, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Ora sei connesso come: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Profilo di %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 msgid "Success! Profile Updated" msgstr "Tutto OK! Profilo aggiornato" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Esiste già un account per questa e-mail." @@ -1458,54 +1464,58 @@ msgstr "Esiste già un account per questa e-mail." msgid "Found no valid gmail.json file with OAuth information" msgstr "Nessun file gmail.json valido con informazioni OAuth trovato" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "Elimina contenuto della cartella temporanea" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s inviato all'E-Reader" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" -msgstr "Il convertitore di libri di Calibre %(tool)s non è stato trovato" +msgstr "Calibre ebook-convert %(tool)s non trovato" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "Formato %(format)s non trovato sul disco" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "La conversione del libro è terminata con un errore sconosciuto" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Errore con il convertitore Kepubify: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "File convertito non trovato o più di un file nella cartella %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Errore nel convertitore: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Si è verificato un errore con Calibre: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Errore nel convertitore: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Converti" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Riconessione al database di Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "E-mail" @@ -1513,30 +1523,26 @@ msgstr "E-mail" msgid "Backing up Metadata" msgstr "Backup dei metadati" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "Elimina contenuto della cartella temporanea" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "Sono state generate %(count)s miniature delle copertine" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Miniature delle copertine" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "Sono state generate {0} miniature delle serie" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Cancellazione della cache delle miniature delle copertine" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Caricamento" @@ -1555,11 +1561,11 @@ msgstr "Nome utente" msgid "Email" msgstr "E-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "Invia all'e-mail dell'eReader" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Amministratore" @@ -1569,8 +1575,8 @@ msgstr "Amministratore" msgid "Password" msgstr "Password" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Download" @@ -1780,13 +1786,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Annulla" @@ -1836,16 +1842,76 @@ msgstr "Ordina secondo la data di pubblicazione, prima i più recenti" msgid "Sort according to publishing date, oldest first" msgstr "Ordina secondo la data di pubblicazione, prima i più vecchi" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "riduci" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Altri di" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Libro %(index)s di %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Lingua" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Editore" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Pubblicato" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Descrizione:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Precedente" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Successivo" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Nessun risultato trovato" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Home" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Cerca nella biblioteca" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Disconnetti" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "Tema normale" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Elimina libro" @@ -1874,99 +1940,107 @@ msgstr "Converti in:" msgid "Convert book" msgstr "Converti libro" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Caricamento in corso..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Chiudi" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Errore" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Caricamento terminato, elaborazione in corso..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Carica formato" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Titolo del libro" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autore" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Etichette" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID della serie" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Data di pubblicazione" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Valutazione" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Descrizione" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identificatori" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Tipo di identificatore" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Valore dell'identificatore" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Rimuovi" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Aggiungi identificatore" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Etichette" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID della serie" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Valutazione" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Recupera la copertina dall'URL (JPEG: l'immagine verrà scaricata e archiviata nel database)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Carica la copertina dal disco locale" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Data di pubblicazione" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Editore" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Lingua" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Sì" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "No" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Carica formato" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Visualizza il libro dopo averlo salvato" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Recupera i metadati" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -1974,37 +2048,31 @@ msgstr "Recupera i metadati" msgid "Save" msgstr "Salva" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Parola chiave" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Cerca parola chiave" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Fai clic sulla copertina per caricare i metadati nel modulo" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Caricamento in corso..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Chiudi" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Origine" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Errore nella ricerca!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nessun risultato trovato! Prova con un'altra parola chiave." @@ -2111,7 +2179,7 @@ msgid "Enter " msgstr "Inserisci" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Sei veramente sicuro?" @@ -2133,7 +2201,7 @@ msgstr "Posizione del database di Calibre" #: cps/templates/config_db.html:21 msgid "Separate Book Files from Library" -msgstr "Separa i file dei libri dalla libreria" +msgstr "Separa i file dei libri dalla biblioteca" #: cps/templates/config_db.html:34 msgid "Use Google Drive?" @@ -2185,7 +2253,7 @@ msgstr "Stabile" #: cps/templates/config_edit.html:46 msgid "Nightly" -msgstr "Notturno" +msgstr "Nightly" #: cps/templates/config_edit.html:50 msgid "Trusted Hosts (Comma Separated)" @@ -2536,7 +2604,7 @@ msgstr "Consenti download" #: cps/templates/config_view_edit.html:96 cps/templates/user_edit.html:105 msgid "Allow eBook Viewer" -msgstr "Consenti utilizzo del visualizzatore di libri" +msgstr "Consenti visualizzatore eBook" #: cps/templates/config_view_edit.html:101 cps/templates/user_edit.html:110 msgid "Allow Uploads" @@ -2583,74 +2651,61 @@ msgstr "Aggiungi etichetta consentiti/negati" msgid "Add Allowed/Denied custom column values" msgstr "Aggiungi valori di colonna personalizzati consentiti/negati" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Leggi nel browser" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Ascolta nel browser" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Libro %(index)s di %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Pubblicato" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Contrassegna come da leggere" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Contrassegna come letto" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" msgstr "Contrassegna il libro come letto o da leggere" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Letto" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Ripristina dall'archivio" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Aggiungi all'archivio" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "Contrassegna il libro come archiviato o no per nasconderlo in Calibre-Web ed eliminarlo da Kobo Reader" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Archive" msgstr "Archivio" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Descrizione:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Aggiungi allo scaffale" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Pubblico)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" -msgstr "Modifica metadati" +msgstr "Modifica i metadati" #: cps/templates/email_edit.html:13 msgid "Email Account Type" @@ -2716,11 +2771,7 @@ msgstr "Inserisci nome del dominio" #: cps/templates/email_edit.html:92 msgid "Denied Domains (Blacklist)" -msgstr "Domini negati (lista nera)" - -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Successivo" +msgstr "Domini non consentiti (lista nera)" #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" @@ -2742,11 +2793,15 @@ msgstr "L'istanza Calibre-Web non è configurata, per favore contatta l'amminist msgid "Create Issue" msgstr "Segnala problema" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +msgid "Return to Database config" +msgstr "Ritorna alla configurazione del database" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Ritorna alla pagina principale" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Disconnetti utente" @@ -2836,7 +2891,7 @@ msgstr "Libri ordinati per valutazione" msgid "Books ordered by file formats" msgstr "Libri ordinati per formato di file" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Scaffali" @@ -2845,60 +2900,36 @@ msgstr "Scaffali" msgid "Books organized in shelves" msgstr "Libri organizzati in scaffali" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Home" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Attiva/disattiva navigazione" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Cerca nella biblioteca" +#: cps/templates/layout.html:59 +msgid "Simple Theme" +msgstr "Tema semplice" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Account" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Disconnetti" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Caricamento in corso..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Errore" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Caricamento terminato, elaborazione in corso..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Impostazioni" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Per favore non aggiornare la pagina" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Naviga" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Informazioni su" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Precedente" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Dettagli del libro" @@ -3014,42 +3045,82 @@ msgstr "Cartella superiore" msgid "Select" msgstr "Seleziona" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "OK" #: cps/templates/osd.xml:5 msgid "Calibre-Web eBook Catalog" -msgstr "Catalogo libri di Calibre-Web" +msgstr "Catalogo eBook Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "Lettore epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +msgid "Choose a theme below:" +msgstr "Scegli un tema qui sotto:" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Chiaro" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Scuro" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Seppia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Nero" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Adatta il testo quando le barre laterali sono aperte." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "Dimensioni caratteri" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "Carattere" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "Predefinito" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "Yahei" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "SimSun" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "KaiTi" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "Doppia pagina" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "Due colonne" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "Una colonna" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Lettore di fumetti" @@ -3226,10 +3297,6 @@ msgstr "Questo link di verifica scadrà tra 10 minuti." msgid "Generate Series Cover Thumbnails" msgstr "Genera miniature delle copertine delle serie" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Nessun risultato trovato" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Termine di ricerca:" @@ -3246,11 +3313,11 @@ msgstr "Data di pubblicazione dal" msgid "Published Date To" msgstr "Data di pubblicazione fino al" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "Qualsiasi" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "Vuoto" @@ -3286,11 +3353,13 @@ msgstr "Valutazione superiore a" msgid "Rating Below" msgstr "Valutazione inferiore a" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Da:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "A:" @@ -3314,6 +3383,14 @@ msgstr "Disabilita la modifica della disposizione" msgid "Enable Change order" msgstr "Abilita la modifica della disposizione" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "Ordina secondo il libro aggiunto allo scaffale, prima il più recente" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "Ordina secondo il libro aggiunto allo scaffale, prima il più vecchio" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Condividi con tutti" @@ -3382,15 +3459,19 @@ msgstr "Avanzamento" msgid "Run Time" msgstr "Tempo d'esecuzione" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "Messaggio" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Azioni" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Questa attività verrà annullata. Tutti i progressi compiuti da questa attività verranno salvati." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Se si tratta di un'attività pianificata, verrà eseguita nuovamente all'orario pianificato successivo." @@ -3398,6 +3479,10 @@ msgstr "Se si tratta di un'attività pianificata, verrà eseguita nuovamente all msgid "Reset user Password" msgstr "Reimposta la password dell'utente" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "Invia all'indirizzo e-mail dell'eReader. Usa la virgola per separare le email per più eReader" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Lingua dei libri" @@ -3484,11 +3569,11 @@ msgstr "Etichette consentite" #: cps/templates/user_table.html:140 msgid "Edit Denied Tags" -msgstr "Modifica le etichette negate" +msgstr "Modifica le etichette non consentite" #: cps/templates/user_table.html:140 msgid "Denied Tags" -msgstr "Etichette negate" +msgstr "Etichette non consentite" #: cps/templates/user_table.html:141 msgid "Edit Allowed Column Values" @@ -3500,11 +3585,11 @@ msgstr "Valori delle colonne consentiti" #: cps/templates/user_table.html:142 msgid "Edit Denied Column Values" -msgstr "Modifica valori di colonna negati" +msgstr "Modifica i valori per le colonne non consentite" #: cps/templates/user_table.html:142 msgid "Denied Column Values" -msgstr "Valori di colonna negati" +msgstr "Valori per le colonne non consentite" #: cps/templates/user_table.html:144 msgid "Change Password" diff --git a/cps/translations/ja/LC_MESSAGES/messages.mo b/cps/translations/ja/LC_MESSAGES/messages.mo index a707f3b7..4db2f956 100644 Binary files a/cps/translations/ja/LC_MESSAGES/messages.mo and b/cps/translations/ja/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/ja/LC_MESSAGES/messages.po b/cps/translations/ja/LC_MESSAGES/messages.po index 348eb099..79b1e1ac 100644 --- a/cps/translations/ja/LC_MESSAGES/messages.po +++ b/cps/translations/ja/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2018-02-07 02:20-0500\n" "Last-Translator: subdiox \n" "Language: ja\n" @@ -16,501 +16,519 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "統計" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "サーバーを再起動しました。ページを再読み込みしてください" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "サーバーをシャットダウンしています。ページを閉じてください" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "不明なコマンド" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "%(email)s へのテストメール送信がキューに追加されました。結果を見るにはタスクを確認してください" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "不明" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "管理者ページ" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "基本設定" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "UI設定" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "カスタムカラムの%(column)d列目がcalibreのDBに存在しません" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "ユーザーを編集" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "全て" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "ユーザーが見つかりません" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{}人のユーザーが削除されました" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "全て表示" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "不正なリクエスト" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "ゲストユーザーの名前は変更できません" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "ゲストユーザーはこのロールを持つことができません" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "管理者ユーザーが残っておらず、管理者ロールを削除できません" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "値はtrueかfalseのどちらかでなければなりません" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "無効なロール" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "ゲストユーザーはこの画面を表示できません" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "無効な表示" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "ゲストユーザーの言語設定は自動的に決定されるため、固定することはできません" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "有効な言語設定がありません" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "有効な本の言語がありません" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "パラメータが見つかりません" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "無効な読み取り列" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "無効な制限列" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Webの設定を更新しました" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Koboのトークンを削除してもよろしいですか?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "このドメインを削除してもよろしいですか?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "このユーザーを削除してもよろしいですか?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "この本棚を削除してもよろしいですか?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "選択したユーザーの言語設定を変更してもよろしいですか?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "選択したユーザーが表示できる本の言語を変更してもよろしいですか?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "選択したユーザーの選択したロールを変更してもよろしいですか?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "選択したユーザーの選択した制限を変更してもよろしいですか?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "選択したユーザーの選択した表示制限を変更してもよろしいですか?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "選択したユーザーの本棚同期の動作を変更してもよろしいですか?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Calibreライブラリのパスを変更してもよろしいですか?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calibre-Webは新しい表紙を検索してそのサムネイルを更新しますが、これにはしばらく時間がかかるかもしれません" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Calibre-Webの同期DBを削除して強制的にKoboリーダーと同期してもよろしいですか?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "拒否" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "許可" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{}件の同期項目を削除しました" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "タグが見つかりません" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "無効なアクションです" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.jsonがWebアプリケーション用に設定されていません" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "ログファイルの場所が無効です。正しいパスを入力してください" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "アクセスログファイルの場所が無効です。正しいパスを入力してください" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "LDAPのプロバイダ、ポート番号、DN、ユーザーIDを入力してください" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "LDAPのサービスアカウント名とパスワードを入力してください" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "LDAPのサービスアカウント名を入力してください" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAPのグループフィルタには \"%s\" というフォーマットのIDが一つ必要です" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAPのグループフィルタ内の括弧が一致しません" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAPのユーザーフィルタには \"%s\" というフォーマットのIDが一つ必要です" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAPのユーザーフィルタ内の括弧が一致しません" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAPのメンバーフィルタには \"%s\" というフォーマットのIDが一つ必要です" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "LDAPのメンバーフィルタ内の括弧が一致しません" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAPのCA証明書、証明書、キーの場所が無効です。正しいパスを入力してください" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "新規ユーザーを追加" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "メールサーバー設定を編集" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "DBエラー: %(error)s" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "%(email)s へのテストメール送信がキューに追加されました。結果を見るにはタスクを確認してください" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "%(res)s へのテストメール送信中にエラーが発生しました" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "初めにメールアドレスを設定してください" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "メールサーバーの設定を更新しました" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "スケジュールタスク設定を編集" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "指定したタスクの開始時刻が無効です" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "指定したタスクの期間が無効です" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "スケジュールタスクの設定を更新しました" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "不明なエラーが発生しました。あとで再試行してください。" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "設定DBが書き込みできません" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "ユーザー %(nick)s を編集" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "ユーザー %(user)s のパスワードをリセット" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "初めにSMTPメールの設定をしてください" -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "ログファイルビューア" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "更新データを要求中" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "更新データをダウンロード中" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "更新データを展開中" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "ファイルを置換中" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "DB接続を切断" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "サーバー停止中" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "アップデート完了、OKを押してページを再読み込みしてください" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "アップデート失敗:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTPエラー" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "接続エラー" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "接続確立中にタイムアウトしました" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "エラー発生" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "更新データを一時フォルダに保存できませんでした" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "更新中にファイルを置換できませんでした" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "少なくとも1人のLDAPユーザーの抽出に失敗しました" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "少なくとも1人のLDAPユーザーの作成に失敗しました" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "エラー: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "エラー: LDAPサーバーのレスポンスでユーザーが返されません" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "DB内にLDAPユーザーが1人も見つかりません" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{}人のユーザーをインポートしました" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "DBの場所が無効です。正しいパスを入力してください" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "DBへの書き込みができません" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "キーファイルの場所が無効です。正しいパスを入力してください" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "証明書ファイルの場所が無効です。正しいパスを入力してください" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "DB設定を更新しました" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "DB設定" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "全ての項目を入力してください" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "このメールは有効なドメインからのものではありません" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "新規ユーザー追加" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "ユーザー '%(user)s' を作成しました" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "このメールアドレスかニックネームで登録されたアカウントがすでに存在します。" -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "ユーザー '%(nick)s' を削除しました" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "ゲストユーザーは削除できません" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "管理者ユーザーが残っておらず、ユーザーを削除できません" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "ユーザー '%(nick)s' を更新しました" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "検索" + #: cps/converter.py:31 msgid "not installed" msgstr "インストールされていません" @@ -519,128 +537,123 @@ msgstr "インストールされていません" msgid "Execution permissions missing" msgstr "実行権限がありません" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "カスタムカラムの%(column)d列目がcalibreのDBに存在しません" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "なし" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "選択した本は利用できません。ファイルが存在しないか、アクセスできません" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "ユーザーは表紙をアップロードする権限がありません" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "IDは大文字小文字を区別しません。元のIDを上書きします" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "メタデータを更新しました" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "本編集中のエラー: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "ファイル %(file)s をアップロードしました" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "変換元の形式または変換後の形式が指定されていません" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "本の %(book_format)s への変換がキューに追加されました" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "この本の変換中にエラーが発生しました: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "アップロードした本はすでにライブラリに存在します。新しくアップロードする前に変更を加えてください: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "選択した本は利用できません。ファイルが存在しないか、アクセスできません" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "ユーザーは表紙をアップロードする権限がありません" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "IDは大文字小文字を区別しません。元のIDを上書きします" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "'%(langname)s' は有効な言語ではありません" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "メタデータを更新しました" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "本編集中のエラー: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "アップロードした本はすでにライブラリに存在します。新しくアップロードする前に変更を加えてください: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "ファイル拡張子 '%(ext)s' をこのサーバーにアップロードすることは許可されていません" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "ファイル拡張子 '%(ext)s' をこのサーバーにアップロードすることは許可されていません" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "アップロードするファイルには拡張子が必要です" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "ファイル %(filename)s は一時フォルダに保存できませんでした" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "表紙ファイル %(file)s の移動に失敗しました: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "本の形式を削除しました" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "本を削除しました" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "本を削除する権限がありません" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "メタデータを編集" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s は有効な数字ではありません。スキップします" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "新たなファイル形式をアップロードする権限がありません" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "%(path)s の作成に失敗しました (Permission denied)。" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "ファイル %(file)s を保存できません。" -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "ファイル形式 %(ext)s が %(book)s に追加されました" @@ -653,485 +666,478 @@ msgstr "Googleドライブの設定が完了していません。Googleドライ msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "コールバックドメインが認証されていません。Google Developer Consoleでドメインを認証してください" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "ID: %(book)d の本に %(format)s フォーマットはありません" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "Googleドライブ: %(fn)s に %(format)s はありません" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s がありません: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "E-Readerに送信" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "このメールはCalibre-Web経由で送信されました。" -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web テストメール" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "テストメール" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Calibre-Webを始める" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "ユーザー: %(name)s 用の登録メール" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "%(orig)s を %(format)s に変換してからE-Readerに送信" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "E-Readerに %(format)s を送信" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s をE-Readerに送信" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "要求されたファイルを読み込めませんでした。権限設定が正しいか確認してください。" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "読み込みステータスを設定できません: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "%(id)s の本フォルダの削除に失敗しました。そこにはサブフォルダがあります: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "本 %(id)s の削除に失敗しました: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "本 %(id)s はDBのみから削除されます。DB内の本のパスが有効ではありません: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "エラー: %(error)s により、著者名を %(src)s から %(dest)s に変更できませんでした" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "ファイル %(file)s はGoogleドライブ上にありません" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "エラー: %(error)s により、タイトルを %(src)s から %(dest)s に変更できませんでした" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "本のパス %(path)s はGoogleドライブ上にありません" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "このユーザー名はすでに使われています" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "メールアドレスの形式が無効" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "表紙のアップロードに必要なPythonモジュール 'advocate' がインストールされていません" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "表紙のダウンロードに失敗しました" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "表紙形式エラー" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "表紙アップロードのためにlocalhostやローカルネットワークにアクセスすることは許可されていません" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "表紙ファイルの作成に失敗しました" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "表紙ファイルが有効な画像ファイルでないか、または保存できませんでした" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "表紙ファイルは jpg/jpeg/png/webp/bmp のみ対応しています" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "表紙ファイルの内容が無効です" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "表紙ファイルは jpg/jpeg のみ対応しています" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "見つける" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "UnRarのバイナリファイルが見つかりません" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "UnRarの実行中にエラーが発生しました" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "DBへの書き込みができません" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "実行権限がありません" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "UnRarの実行中にエラーが発生しました" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "localhost以外からCalibre-Webにアクセスし、有効なKobo端末用APIエンドポイントを取得してください" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo設定" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "%(provider)s で登録" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "%(nickname)s としてログイン中" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "%(oauth)s との連携に成功しました" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "ログイン失敗、OAuthアカウントと連携しているユーザーが存在しません" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "%(oauth)s との連携解除に成功しました" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "%(oauth)s との連携解除に失敗しました" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "%(oauth)s と連携していません" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "GitHubアカウントでのログインに失敗しました。" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "GitHubからのユーザー情報取得に失敗しました。" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Googleアカウントでのログインに失敗しました。" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Googleからのユーザー情報取得に失敗しました。" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub OAuth エラー、再度お試しください。" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "GitHub OAuth エラー: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google OAuth エラー、再度お試しください。" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google OAuth エラー: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "星{}" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "ログイン" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "トークンが見つかりません" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "トークンが無効です" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "成功です!端末に戻ってください" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "本" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "最近追加された本を表示" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "人気の本" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "人気の本を表示" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "ダウンロードされた本" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "ダウンロードされた本を表示" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "高評価の本" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "高評価の本を表示" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "既読の本" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "既読の本と未読の本を表示" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "未読の本" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "未読の本を表示" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "見つける" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "ランダムに本を表示" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "カテゴリ" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "カテゴリ選択を表示" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "シリーズ" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "シリーズ選択を表示" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "著者" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "著者選択を表示" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "出版社" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "出版社選択を表示" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "言語" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "言語選択を表示" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "評価" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "評価選択を表示" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "ファイル形式" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "ファイル形式選択を表示" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "アーカイブされた本" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "アーカイブされた本を表示" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "本の一覧" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "本の一覧を表示" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "検索" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "これ以降に出版 " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "これ以前に出版 " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "評価 <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "評価 >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "既読/未読状況 = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "カスタムカラムの検索でエラーが発生しました。Calibre-Webを再起動してください" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "詳細検索" @@ -1186,7 +1192,7 @@ msgstr "本が %(sname)s から削除されました" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "申し訳ありませんが、あなたはこの本棚から本を削除することが許可されていません" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "本棚を作成する" @@ -1239,45 +1245,45 @@ msgstr "'%(title)s' という名前のみんなの本棚はすでに存在しま msgid "A private shelf with the name '%(title)s' already exists." msgstr "'%(title)s' という名前の本棚はすでに存在します。" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "本棚: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "本棚を開けません。この本棚は存在しないかアクセスできません" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "タスク" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "待機中" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "失敗" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "開始" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "終了" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "終了" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "キャンセル" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "不明" @@ -1310,178 +1316,178 @@ msgstr "アップデートが利用可能です。下のボタンをクリック msgid "No release information available" msgstr "リリース情報がありません" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "本を見つける (ランダムに表示)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "人気の本 (最もダウンロードされた本)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "%(user)s がダウンロードした本" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "著者: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "出版社: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "シリーズ: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "評価: なし" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "評価: 星%(rating)s" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "ファイル形式: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "カテゴリ: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "言語: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "ダウンロード数" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "評価一覧" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "ファイル形式一覧" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "初めにSMTPメールの設定をしてください" -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "本の %(eReadermail)s への送信がキューに追加されました" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "%(res)s を送信中にエラーが発生しました" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "初めにKindleのメールアドレスを設定してください" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "登録" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "メールサーバーが設定されていません。管理者に連絡してください" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "メールサーバーが設定されていません。管理者に連絡してください" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "このメールアドレスは登録が許可されていません" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "確認メールがこのメールアドレスに送信されました。" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "LDAP認証を有効化できません" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "%(nickname)s としてログイン中" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "代わりに '%(nickname)s' としてログインします。LDAPサーバーにアクセスできないか、ユーザーが存在しません" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "ログインできません: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "ユーザー名またはパスワードが違います" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "新しいパスワードがあなたのメールアドレスに送信されました" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "不明なエラーが発生しました。あとで再試行してください。" -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "パスワードをリセットするには、有効なユーザー名を入力してください" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "%(nickname)s としてログイン中" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s のプロフィール" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "プロフィールを更新しました" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "このメールアドレスで登録されたアカウントがすでに存在します" @@ -1489,54 +1495,58 @@ msgstr "このメールアドレスで登録されたアカウントがすでに msgid "Found no valid gmail.json file with OAuth information" msgstr "OAuth情報を含んだ有効なgmail.jsonファイルが見つかりません" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s をE-Readerに送信" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre ebook-convert %(tool)s が見つかりません" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s 形式は存在しません" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Ebook converter が不明なエラーで失敗しました" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-converter が失敗しました: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "変換されたファイルが見つからないか、またはフォルダー %(folder)s 内に複数存在します" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ebook-converter が失敗しました: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre が失敗しました: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ebook-converter が失敗しました: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "変換" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Calibre DBと再接続中" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "メール" @@ -1545,30 +1555,26 @@ msgstr "メール" msgid "Backing up Metadata" msgstr "メタデータを編集" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "表紙サムネイルを%(count)s個生成しました" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "表紙サムネイル" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "シリーズのサムネイルを{0}個生成しました" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "表紙サムネイルのキャッシュを消去中" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "アップロード" @@ -1587,11 +1593,11 @@ msgstr "ユーザー名" msgid "Email" msgstr "メールアドレス" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "E-Readerメールアドレス" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "管理者" @@ -1601,8 +1607,8 @@ msgstr "管理者" msgid "Password" msgstr "パスワード" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "ダウンロード" @@ -1813,13 +1819,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "キャンセル" @@ -1869,16 +1875,76 @@ msgstr "発売日が新しい順にソート" msgid "Sort according to publishing date, oldest first" msgstr "発売日が古い順にソート" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "減らす" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "同じ著者が書いた本: " +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "%(range)s 第%(index)s巻" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "言語" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "出版社" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "発売日" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "詳細:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "前" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "次" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "結果が見つかりません" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "ホーム" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "ライブラリ内を検索" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "ログアウト" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "本を削除" @@ -1907,99 +1973,107 @@ msgstr "変換先:" msgid "Convert book" msgstr "本を変換" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "アップロード中..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "閉じる" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "エラー" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "アップロード完了。現在処理中ですのでお待ち下さい..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "アップロード形式" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "本のタイトル" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "著者" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "タグ" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "巻数" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "発売日" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "評価" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "詳細" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "識別子" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "識別子タイプ" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "識別子の値" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "削除" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "識別子を追加" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "タグ" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "巻数" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "評価" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "表紙をURLから取得 (JPEG画像がダウンロードされDBに保存されます)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "表紙をアップロード" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "発売日" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "出版社" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "言語" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "既読" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "未読" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "アップロード形式" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "保存後に本を表示" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "メタデータを取得" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2007,37 +2081,31 @@ msgstr "メタデータを取得" msgid "Save" msgstr "保存" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "キーワード" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "キーワード検索" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "カバー画像をクリックしてメタデータをフォームに読み込んでください" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "読み込み中..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "閉じる" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "ソース" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "検索エラー" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "検索結果が見つかりません。別のキーワードで検索してみてください。" @@ -2144,7 +2212,7 @@ msgid "Enter " msgstr "入力: " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "よろしいですか?" @@ -2620,74 +2688,61 @@ msgstr "許可/拒否タグを追加" msgid "Add Allowed/Denied custom column values" msgstr "許可/拒否カスタムカラムを追加" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "ブラウザで読む" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "ブラウザで聞く" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "%(range)s 第%(index)s巻" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "発売日" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "未読に設定" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "既読に設定" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "未読に設定" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "既読" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "アーカイブから復元" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "アーカイブに追加" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "アーカイブ済み" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "詳細:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "本棚に追加" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(公開)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "メタデータを編集" @@ -2760,10 +2815,6 @@ msgstr "ドメイン名を入力" msgid "Denied Domains (Blacklist)" msgstr "拒否するドメイン名 (ブラックリスト)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "次" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr ".kobo/Kobo/Kobo eReader.confファイルをテキストエディタで開いて追加(編集)してください:" @@ -2784,11 +2835,16 @@ msgstr "Calibre-Webインスタンスが未設定です。管理者に連絡し msgid "Create Issue" msgstr "Issueを作成" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "DB設定" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "ホームに戻る" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "ユーザーをログアウト" @@ -2878,7 +2934,7 @@ msgstr "評価順" msgid "Books ordered by file formats" msgstr "ファイル形式順" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "本棚" @@ -2887,60 +2943,37 @@ msgstr "本棚" msgid "Books organized in shelves" msgstr "本棚に整理された本" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "ホーム" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "ナビゲーションを実行" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "ライブラリ内を検索" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "簡単" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "アカウント" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "ログアウト" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "アップロード中..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "エラー" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "アップロード完了。現在処理中ですのでお待ち下さい..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "設定" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "ページを更新しないでください" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "閲覧" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "このサイトについて" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "前" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "本の詳細" @@ -3056,7 +3089,7 @@ msgstr "親ディレクトリ" msgid "Select" msgstr "選択" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "OK" @@ -3064,34 +3097,80 @@ msgstr "OK" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-WebのeBookカタログ" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "EPUBリーダー" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "ユーザー名を入力してください" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "ライト" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "ダーク" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "セピア" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "ブラック" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "サイドバーが開いているとき、テキストを再度流し込みます。" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "削除" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "待機中" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "垂直方向" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "既読" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "無効な読み取り列" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "コミックリーダー" @@ -3269,10 +3348,6 @@ msgstr "この確認リンクの有効期限は10分です。" msgid "Generate Series Cover Thumbnails" msgstr "シリーズの表紙サムネイルを生成" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "結果が見つかりません" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "検索する単語:" @@ -3289,11 +3364,11 @@ msgstr "発売日(〜から)" msgid "Published Date To" msgstr "発売日(〜まで)" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3329,11 +3404,13 @@ msgstr "評価(〜以上)" msgid "Rating Below" msgstr "評価(〜以下)" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "〜から:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "〜まで:" @@ -3357,6 +3434,16 @@ msgstr "並び順の変更を無効にする" msgid "Enable Change order" msgstr "並び順の変更を有効にする" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "追加日が新しい順にソート" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "追加日が古い順にソート" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "みんなと共有" @@ -3425,15 +3512,20 @@ msgstr "進捗" msgid "Run Time" msgstr "稼働時間" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "統合" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "アクション" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "このタスクはキャンセルされます。このタスクによる進捗はすべて保存されます。" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "これがスケジュールタスクの場合、次のスケジュールの時刻にはもう一度実行されます。" @@ -3441,6 +3533,10 @@ msgstr "これがスケジュールタスクの場合、次のスケジュール msgid "Reset user Password" msgstr "パスワードをリセット" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "本の言語" diff --git a/cps/translations/km/LC_MESSAGES/messages.mo b/cps/translations/km/LC_MESSAGES/messages.mo index 4ccbaee3..809d81bd 100644 Binary files a/cps/translations/km/LC_MESSAGES/messages.mo and b/cps/translations/km/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/km/LC_MESSAGES/messages.po b/cps/translations/km/LC_MESSAGES/messages.po index 593a9e2e..ec5b6f6d 100644 --- a/cps/translations/km/LC_MESSAGES/messages.po +++ b/cps/translations/km/LC_MESSAGES/messages.po @@ -8,7 +8,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2018-08-27 17:06+0700\n" "Last-Translator: \n" "Language: km_KH\n" @@ -17,508 +17,526 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "ស្ថិតិ" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "ម៉ាស៊ីន server បានដំណើរការម្តងទៀត សូមបើកទំព័រជាថ្មី" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "កំពុងបិទម៉ាស៊ីន server សូមបិទផ្ទាំងនេះ" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "សៀវភៅបានចូលជួរសម្រាប់ផ្ញើទៅ %(eReadermail)s ដោយជោគជ័យ" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "មិនដឹង" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "ទំព័ររដ្ឋបាល" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "ការកំណត់សាមញ្ញ" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "ការកំណត់ផ្ទាំងប្រើប្រាស់" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "អ្នកប្រើប្រាស់រដ្ឋបាល" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "បង្ហាញទាំងអស់" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "តើអ្នកពិតជាចង់លុបធ្នើនេះមែនទេ?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "តើអ្នកពិតជាចង់លុបធ្នើនេះមែនទេ?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "តើអ្នកពិតជាចង់លុបធ្នើនេះមែនទេ?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "តើអ្នកពិតជាចង់លុបធ្នើនេះមែនទេ?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "តើអ្នកពិតជាចង់លុបធ្នើនេះមែនទេ?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "ប្តូរការកំណត់ SMTP" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "កែប្រែអ្នកប្រើប្រាស់ %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "សូមកំណត់អ៊ីមែល SMTP ជាមុនសិន" -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "កំពុងស្នើសុំឯកសារបច្ចុប្បន្នភាព" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "កំពុងទាញយកឯកសារបច្ចុប្បន្នភាព" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "កំពុងពន្លាឯកសារបច្ចុប្បន្នភាព" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "ទំនាក់ទំនងទៅមូលដ្ឋានទិន្នន័យត្រូវបានផ្តាច់" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "ការធ្វើបច្ចុប្បន្នភាពបានបញ្ចប់ សូមចុច okay រួចបើកទំព័រជាថ្មី" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "ទំនាក់ទំនងទៅមូលដ្ឋានទិន្នន័យត្រូវបានផ្តាច់" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "ការកំណត់មុខងារ" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "សូមបំពេញចន្លោះទាំងអស់!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "បន្ថែមអ្នកប្រើប្រាស់ថ្មី" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "បានបង្កើតអ្នកប្រើប្រាស់ ‘%(user)s’" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "" -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "អ្នកប្រើប្រាស់ ‘%(nick)s’ ត្រូវបានលុប" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "អ្នកប្រើប្រាស់ ‘%(nick)s’ ត្រូវបានកែប្រែ" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "ស្វែងរក" + #: cps/converter.py:31 msgid "not installed" msgstr "មិនបានតម្លើង" @@ -527,128 +545,123 @@ msgstr "មិនបានតម្លើង" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "គ្មាន" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "ឯកសារប្រភេទ '%(ext)s' មិនត្រូវបានអនុញ្ញាតឲអាប់ឡូដទៅម៉ាស៊ីន server នេះទេ" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "ឯកសារប្រភេទ '%(ext)s' មិនត្រូវបានអនុញ្ញាតឲអាប់ឡូដទៅម៉ាស៊ីន server នេះទេ" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "ឯកសារដែលត្រូវអាប់ឡូដត្រូវមានកន្ទុយឯកសារ" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "កែប្រែទិន្នន័យមេតា" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "មិនអាចបង្កើតទីតាំង %(path)s (ពុំមានសិទ្ធិ)។" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "មិនអាចរក្សាទុកឯកសារ %(file)s ។" -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "ឯកសារទម្រង់ %(ext)s ត្រូវបានបន្ថែមទៅ %(book)s" @@ -661,479 +674,472 @@ msgstr "" msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Callback domain មិនទាន់បានផ្ទៀងផ្ទាត់ឲប្រើទេ សូមធ្វើតាមជំហានដើម្បីផ្ទៀងផ្ទាត់ domain នៅក្នុង Google Developer Console" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "ផ្ញើទៅ Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "" -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "ផ្ញើទៅ Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "ផ្ញើទៅ Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "ឯកសារដែលបានស្នើសុំមិនអាចបើកបានទេ។ អាចនឹងខុសសិទ្ធិប្រើប្រាស់ទេដឹង?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "ប្តូរចំណងជើងពី “%(src)s” ទៅជា “%(dest)s” បរាជ័យដោយបញ្ហា: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "ឯកសារ %(file)s រកមិនឃើញក្នុង Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "ប្តូរចំណងជើងពី “%(src)s” ទៅជា “%(dest)s” បរាជ័យដោយបញ្ហា: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "ទីតាំងសៀវភៅ %(path)s រកមិនឃើញក្នុង Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "ស្រាវជ្រាវ" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "ឥឡូវអ្នកបានចូលដោយមានឈ្មោះថា៖ ‘%(nickname)s’" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "ចូលប្រើប្រាស់" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "រកមិនឃើញវត្ថុតាង" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "វត្ថុតាងហួសពេលកំណត់" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "ជោគជ័យ! សូមវិលមកឧបករណ៍អ្នកវិញ" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "បង្ហាញសៀវភៅមកថ្មី" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "សៀវភៅដែលមានប្រជាប្រិយភាព" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "បង្ហាញសៀវភៅដែលមានប្រជាប្រិយភាព" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "សៀវភៅដែលមានការវាយតម្លៃល្អជាងគេ" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "បង្ហាញសៀវភៅដែលមានការវាយតម្លៃល្អជាងគេ" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "សៀវភៅដែលបានអានរួច" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "បង្ហាញអានរួច និងមិនទាន់អាន" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "សៀវភៅដែលមិនទាន់បានអាន" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "ស្រាវជ្រាវ" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "បង្ហាញសៀវភៅចៃដន្យ" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "ប្រភេទនានា" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "បង្ហាញជម្រើសប្រភេទ" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "ស៊េរី" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "បង្ហាញជម្រើសស៊េរី" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "អ្នកនិពន្ធ" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "បង្ហាញជម្រើសអ្នកនិពន្ធ" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "បង្ហាញជម្រើសស៊េរី" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "ភាសានានា" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "បង្ហាញផ្នែកភាសា" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "បង្ហាញជម្រើសស៊េរី" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "បង្ហាញជម្រើសស៊េរី" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "បង្ហាញសៀវភៅមកថ្មី" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "ស្វែងរក" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "បានបោះពុម្ភក្រោយ " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "បានបោះពុម្ភមុន " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "ការវាយតម្លៃ <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "ការវាយតម្លៃ >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "ស្វែងរកកម្រិតខ្ពស់" @@ -1189,7 +1195,7 @@ msgstr "សៀវភៅត្រូវបានដកចេញពីធ្នើ msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "បង្កើតធ្នើ" @@ -1243,45 +1249,45 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "ធ្នើ៖ ‘%(name)s’" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "មានបញ្ហាពេលបើកធ្នើ។ ពុំមានធ្នើ ឬមិនអាចបើកបាន" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "កិច្ចការនានា" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "កំពុងរង់ចាំ" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "បានបរាជ័យ" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "បានចាប់ផ្តើម" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "បានបញ្ចប់" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "" @@ -1314,173 +1320,173 @@ msgstr "" msgid "No release information available" msgstr "" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "ស្រាវជ្រាវ (សៀវភៅចៃដន្យ)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "សៀវភៅដែលត្រូវបានទាញយកច្រើនជាងគេ" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "ស៊េរី៖ %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "ប្រភេទ៖ %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "ភាសា៖ %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "ឯកសារ DLS" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "សូមកំណត់អ៊ីមែល SMTP ជាមុនសិន" -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "សៀវភៅបានចូលជួរសម្រាប់ផ្ញើទៅ %(eReadermail)s ដោយជោគជ័យ" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "មានបញ្ហានៅពេលផ្ញើសៀវភៅនេះ៖ %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "ចុះឈ្មោះ" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "ឥឡូវអ្នកបានចូលដោយមានឈ្មោះថា៖ ‘%(nickname)s’" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "ខុសឈ្មោះអ្នកប្រើប្រាស់ ឬលេខសម្ងាត់" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "" -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "ខុសឈ្មោះអ្នកប្រើប្រាស់ ឬលេខសម្ងាត់" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "ឥឡូវអ្នកបានចូលដោយមានឈ្មោះថា៖ ‘%(nickname)s’" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "ព័ត៌មានសង្ខេបរបស់ %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "ព័ត៌មានសង្ខេបបានកែប្រែ" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "" @@ -1488,54 +1494,58 @@ msgstr "" msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "ផ្ញើទៅ Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ebook-converter បានបរាជ័យ៖ %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ebook-converter បានបរាជ័យ៖ %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1544,30 +1554,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "កែប្រែទិន្នន័យមេតា" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "អាប់ឡូដ" @@ -1586,12 +1592,12 @@ msgstr "ឈ្មោះហៅក្រៅ" msgid "Email" msgstr "" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "ឧបករណ៍ Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "រដ្ឋបាល" @@ -1601,8 +1607,8 @@ msgstr "រដ្ឋបាល" msgid "Password" msgstr "លេខសម្ងាត់" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "ទាញយក" @@ -1813,13 +1819,13 @@ msgid "OK" msgstr "បាទ/ចាស" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "" @@ -1869,16 +1875,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "បន្ថែមទៀតដោយ" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "ភាសា" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "អ្នកបោះពុម្ភ" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "ពិពណ៌នា" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "មុន" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "បន្ទាប់" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "ចេញពីការប្រើប្រាស់" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "លុបសៀវភៅ" @@ -1907,99 +1973,107 @@ msgstr "" msgid "Convert book" msgstr "" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "កំពុងអាប់ឡូត..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "បិទ" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "" + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "ទម្រង់អាប់ឡូដ" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "ចំណងជើងសៀវភៅ" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "អ្នកនិពន្ធ" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Tag" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "លេខសម្គាល់ស៊េរី" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "ថ្ងៃបោះពុម្ភ" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "ការវាយតម្លៃ" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "ពិពណ៌នា" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Tag" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "លេខសម្គាល់ស៊េរី" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "ការវាយតម្លៃ" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "URL របស់ក្របមុខ (ឯកសារ JPG ក្របមុខត្រូវបានទាញយក និងរក្សាទុកក្នុង database ក្រោយមកចន្លោះនេះទទេម្តងទៀត)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "ថ្ងៃបោះពុម្ភ" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "អ្នកបោះពុម្ភ" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "ភាសា" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "បាទ/ចាស" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "ទេ" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "ទម្រង់អាប់ឡូដ" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "មើលសៀវភៅក្រោយពីកែប្រែ" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "មើលទិន្នន័យមេតា" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2007,38 +2081,32 @@ msgstr "មើលទិន្នន័យមេតា" msgid "Save" msgstr "" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "ពាក្យគន្លឹះ" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr "ស្វែងរកពាក្យគន្លឹះ" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "ចុចលើគម្របដើម្បីបញ្ចូលទិន្នន័យមេតាទៅក្នុង form" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "កំពុងដំណើរការ..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "បិទ" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "ប្រភព" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "ការស្វែងរកមានកំហុស!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "" @@ -2146,7 +2214,7 @@ msgid "Enter " msgstr "ចុះឈ្មោះ" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "" @@ -2620,73 +2688,60 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "អានក្នុងកម្មវិធី browser" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" msgstr "" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "អាន" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "ស្វែងរក" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "ពិពណ៌នា" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "បន្ថែមទៅធ្នើ" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "កែប្រែទិន្នន័យមេតា" @@ -2757,10 +2812,6 @@ msgstr "" msgid "Denied Domains (Blacklist)" msgstr "" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "បន្ទាប់" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2781,11 +2832,16 @@ msgstr "" msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "ការកំណត់មុខងារ" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2875,7 +2931,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2884,60 +2940,36 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "បិទ/បើកការរុករក" -#: cps/templates/layout.html:47 -msgid "Search Library" +#: cps/templates/layout.html:59 +msgid "Simple Theme" msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "ចេញពីការប្រើប្រាស់" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "កំពុងអាប់ឡូត..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "" - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "ការកំណត់" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "រុករក" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "អំពី" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "មុន" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "ព័ត៌មានលម្អិតរបស់សៀវភៅ" @@ -3053,7 +3085,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "សៀវភៅ" @@ -3062,35 +3094,79 @@ msgstr "សៀវភៅ" msgid "Calibre-Web eBook Catalog" msgstr "" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "ជ្រើសរើសឈ្មោះអ្នកប្រើប្រាស់" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "មកក្រោយ" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "សេរេអត្ថបទនៅពេលបើកផ្ទាំងចំហៀង។" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "លុប" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "កំពុងរង់ចាំ" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "អាន" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "" @@ -3269,10 +3345,6 @@ msgstr "" msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "" @@ -3289,11 +3361,11 @@ msgstr "ថ្ងៃបោះពុម្ភចាប់ពី" msgid "Published Date To" msgstr "ថ្ងៃបោះពុម្ភរហូតដល់" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3330,11 +3402,13 @@ msgstr "ការវាយតម្លៃលើសពី" msgid "Rating Below" msgstr "ការវាយតម្លៃតិចជាង" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3358,6 +3432,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "តើធ្នើនេះគួរជាសាធារណៈទេ?" @@ -3426,15 +3508,19 @@ msgstr "ដំណើរការ" msgid "Run Time" msgstr "រយៈពេលដែលបានចាប់ផ្តើម" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3442,6 +3528,10 @@ msgstr "" msgid "Reset user Password" msgstr "" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "បង្ហាញសៀវភៅដែលមានភាសា" diff --git a/cps/translations/ko/LC_MESSAGES/messages.mo b/cps/translations/ko/LC_MESSAGES/messages.mo index f7170aea..d5429bff 100644 Binary files a/cps/translations/ko/LC_MESSAGES/messages.mo and b/cps/translations/ko/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/ko/LC_MESSAGES/messages.po b/cps/translations/ko/LC_MESSAGES/messages.po index e98bf6aa..2087879e 100644 --- a/cps/translations/ko/LC_MESSAGES/messages.po +++ b/cps/translations/ko/LC_MESSAGES/messages.po @@ -1,1576 +1,1548 @@ -# Korean translation for Calibre-web -# Copyright (C) 2022 EPUBGUIDE.NET -# This file is distributed under the same license as the Calibre-web project -# First korean translation by 내맘대로의 EPUBGUIDE.NET +# Korean translation for Calibre-Web +# Copyright (C) 2024 limeade23 +# This file is distributed under the same license as the Calibre-Web +# First Author , 2022. +# msgid "" msgstr "" -"Project-Id-Version: Calibre-web\n" +"Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/calibre-web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" -"PO-Revision-Date: 2022-01-10 11:30+0900\n" -"Last-Translator: 내맘대로의 EPUBGUIDE.NET \n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-11-01 17:50+0900\n" +"Last-Translator: limeade23 \n" "Language: ko\n" -"Language-Team: 내맘대로의 epubguide.net \n" -"Plural-Forms: nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);\n" +"Language-Team: Korean <>\n" +"Plural-Forms: nplurals=1; plural=0;\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "통계" -#: cps/admin.py:150 -#, fuzzy +#: cps/admin.py:151 msgid "Server restarted, please reload page." -msgstr "서버 다시 시작으로 새로고침 필요" +msgstr "서버가 재시작 되었습니다. 페이지를 새로고침 해주세요." -#: cps/admin.py:152 -#, fuzzy +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." -msgstr "서버를 종료하는 중, 창을 닫아야 함" +msgstr "서버를 종료하고 있습니다. 창을 닫아주세요." -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" -msgstr "성공적으로 DB를 다시 연결하였습니다." +msgstr "성공적으로 데이터베이스를 다시 연결했습니다." -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" -msgstr "알 수 없는 명령" +msgstr "알 수 없는 명령입니다." -#: cps/admin.py:174 -#, fuzzy +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" -msgstr "%(email)s에 테스트를 위한 이메일을 보냄. 결과 확인 필요" +msgstr "메타데이터 백업이 작업에 추가되었습니다. 작업 결과를 확인해주세요." -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "알 수 없음" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "관리자 페이지" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "기본 설정" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "UI 설정" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "사용자 정의 열 번호 %(column)d이(가) calibre 데이터베이스에 없습니다." + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "사용자 관리" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "모두" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" -msgstr "사용자를 찾을 수 없음" +msgstr "사용자를 찾을 수 없습니다." -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" -msgstr "{} 사용자를 성공적으로 삭제함" +msgstr "{} 사용자를 삭제했습니다." -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "모두 보기" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" -msgstr "잘못된 요청" +msgstr "잘못된 요청입니다." -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" -msgstr "Guest 이름은 수정할 수 없음" +msgstr "게스트 이름은 변경할 수 없습니다." -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" -msgstr "Guest는 이 권한을 사용할 수 없음" +msgstr "게스트는 이 권한을 가질 수 없습니다." -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" -msgstr "관리자 계정이 하나 뿐일 때는 관리자를 삭제할 수 없음" +msgstr "남아있는 관리자가 없어 권한을 삭제할 수 없습니다." -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" -msgstr "값으로 rue 또는 false만 설정 가능" +msgstr "true 또는 false여야 합니다." -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" -msgstr "잘못된 권한" +msgstr "잘못된 권한입니다." -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" -msgstr "Guest는 이 view를 사용할 수 없음" +msgstr "게스트는 사용할 수 없습니다." -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" -msgstr "잘못된 view" +msgstr "잘못된 설정입니다." -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" -msgstr "Guest의 로케일은 자동으로 결정되며 설정할 수 없음" +msgstr "게스트의 언어는 자동으로 설정되며 변경할 수 없습니다." -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" -msgstr "유효한 로케일이 아님" +msgstr "올바른 언어가 아닙니다." -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" -msgstr "제공된 책의 언어가 유효하지 않음" +msgstr "제공된 책의 언어가 올바르지 않습니다." -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" -msgstr "매개변수를 찾을 수 없음" - -#: cps/admin.py:572 -msgid "Invalid Read Column" -msgstr "잘못된 읽기 열" +msgstr "매개변수를 찾을 수 없습니다." #: cps/admin.py:578 +msgid "Invalid Read Column" +msgstr "읽기 항목이 유효하지 않습니다." + +#: cps/admin.py:584 msgid "Invalid Restricted Column" -msgstr "잘못된 제한된 열" +msgstr "제한된 항목이 유효하지 않습니다." -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" -msgstr "Calibre-Web 설정이 업데이트 됨" +msgstr "Calibre-Web 설정이 업데이트되었습니다." -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" -msgstr "Kobo Token을 삭제하시겠습니까?" +msgstr "Kobo 토큰을 삭제하시겠습니까?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "이 도메인을 삭제하시겠습니까?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "이 사용자를 삭제하시겠습니까?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "이 서재를 삭제하시겠습니까?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "선택한 사용자의 언어를 변경하시겠습니까?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "선택한 사용자에 대해 표시되는 책 언어를 변경하시겠습니까?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" -msgstr "선택한 사용자에 대해 선택한 권한을 변경하시겠습니까?" +msgstr "선택한 사용자의 권한을 변경하시겠습니까?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" -msgstr "선택한 사용자에 대해 선택한 제한을 변경하시겠습니까?" +msgstr "선택한 사용자의 제한을 변경하시겠습니까?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" -msgstr "선택한 사용자에 대해 선택한 가시성 제한을 변경하시겠습니까?" +msgstr "선택한 사용자의 보기 제한을 변경하시겠습니까?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" -msgstr "선택한 사용자의 실행기 동기화 동작을 변경하시겠습니까?" +msgstr "선택한 사용자의 책장 동기화 설정을 변경하시겠습니까?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" -msgstr "캘리버 서재의 언어를 변경하시겠습니까?" +msgstr "Calibre 라이브러리 경로를 변경하시겠습니까?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" -msgstr "Calibre-Web은 업데이트된 표지를 검색하고 표지 섬네일 업데이트합니다. 시간이 오래 걸릴 수 있습니다." +msgstr "Calibre-Web이 업데이트된 표지를 검색하고 표지 미리보기를 업데이트합니다. 시간이 걸릴 수 있습니다." -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" -msgstr "Kobo Reader와 전체 동기화를 강제 실행하기 위해 Calibre-Web의 동기화 데이터베이스를 삭제하시겠습니까?" +msgstr "Kobo Reader와 전체 동기화를 강제하기 위해 Calibre-Web의 동기화 데이터베이스를 삭제하시겠습니까?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" -msgstr "거부됨" +msgstr "거부" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" -msgstr "허용됨" +msgstr "허용" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" -msgstr "{} 동기화 항목이 삭제됨" +msgstr "{} 동기화 항목이 삭제되었습니다." -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" -msgstr "태그를 찾을 수 없음" +msgstr "태그를 찾을 수 없습니다." -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" -msgstr "잘못된 액션" +msgstr "잘못된 작업입니다." -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" -msgstr "client_secrets.json이 Web 응용프로그램에 대해 설정되지 않음" +msgstr "client_secrets.json이 설정되지 않았습니다." -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" -msgstr "로그 파일 위치가 오류. 올바른 경로 입력 필요" +msgstr "올바른 로그 파일의 경로를 입력해주세요." -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" -msgstr "액세스 로그 파일 위치가 올바르지 않음. 올바른 경로 입력 필요" +msgstr "올바른 액섹스 로그 파일의 경로를 입력해주세요." -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" -msgstr "LDAP 공급자, 포트, DN 및 사용자 개체 식별자를 입력" +msgstr "LDAP 공급자, 포트, DN 및 사용자 개체 식별자를 입력해주세요." -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" -msgstr "LDAP 서비스 계정 및 비밀번호를 입력하십시오" +msgstr "LDAP 서비스 계정과 비밀번호를 입력해주세요." -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" -msgstr "LDAP 서비스 계정을 입력하십시오" +msgstr "LDAP 서비스 계정을 입력해주세요." -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" -msgstr "LDAP 그룹 개체 필터에는 하나의 \"%s\" 형식 식별자가 필요함" +msgstr "LDAP 그룹 개체 필터에는 하나의 \"%s\" 형식 식별자가 필요합니다." -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" -msgstr "LDAP 그룹 개체 필터에 일치하지 않는 괄호가 있음" +msgstr "LDAP 그룹 개체 필터에 맞지 않는 괄호가 있습니다." -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" -msgstr "LDAP 사용자 개체 필터에는 하나의 \"%s\" 형식 식별자 필요" +msgstr "LDAP 사용자 개체 필터에는 하나의 \"%s\" 형식 식별자가 필요합니다." -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" -msgstr "LDAP 사용자 개체 필터에 일치하지 않는 괄호가 있음" +msgstr "LDAP 사용자 개체 필터에 맞지 않는 괄호가 있습니다." -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" -msgstr "LDAP 구성원 사용자 필터에는 하나의 \"%s\" 형식 식별자 필요" +msgstr "LDAP 구성원 사용자 필터에는 하나의 \"%s\" 형식 식별자가 필요합니다." -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" -msgstr "LDAP 구성원 사용자 필터에 일치하지 않는 괄호가 있음" +msgstr "LDAP 구성원 사용자 필터에 맞지 않는 괄호가 있습니다." -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" -msgstr "유요하지 않은 LDAP CACertificate, 인증서 또는 키 위치. 올바른 경로를 입력 필요" +msgstr "LDAP CA인증서, 인증서 또는 키 경로가 유효하지 않습니다." -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" -msgstr "사용자 추가" +msgstr "사용자 생성" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" -msgstr "이메일 서버 설정 편집" +msgstr "이메일 서버 설정" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." -msgstr "Gmail 계정 인증에 성공하였습니다." +msgstr "Gmail 계정 인증에 성공했습니다." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." -msgstr "데이터베이스 오류: %(error)s." +msgstr "데이터베이스 오류가 발생했습니다: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" -msgstr "%(email)s에 테스트를 위한 이메일을 보냄. 결과 확인 필요" +msgstr "%(email)s로 테스트 이메일을 전송했습니다." -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" -msgstr "테스트 이메일을 보내는 동안 오류가 발생했습니다: %(res)s" +msgstr "테스트 이메일 전송 중 오류가 발생했습니다: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." -msgstr "먼저 이메일 주소를 구성하십시오..." +msgstr "이메일 주소를 설정해주세요." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" -msgstr "이메일 서버 설정 업데이트됨" +msgstr "이메일 서버 설정이 업데이트되었습니다." -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" -msgstr "예약 작업 설정 편집" +msgstr "예약 작업 설정" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" -msgstr "지정된 작업의 시작 시간이 잘못 설정되었습니다." +msgstr "작업 시작 시간이 올바르지 않습니다." -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" -msgstr "지정된 작업의 기간이 잘못 설정되었습니다." +msgstr "작업 종료 시간이 올바르지 않습니다." -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" -msgstr "예약된 작업 설정을 업데이트 하였습니다." +msgstr "예약 작업 설정이 업데이트되었습니다." -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." -msgstr "알 수없는 오류가 발생했습니다. 나중에 다시 시도 해주십시오." +msgstr "알 수 없는 오류가 발생했습니다. 잠시 후 다시 시도해 주세요." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" -msgstr "저장할 수 없는 설정 DB입니다." +msgstr "설정 데이터베이스에 저장할 수 없습니다." -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" -msgstr "사용자 %(nick)s 편집" +msgstr "사용자 %(nick)s 수정" -#: cps/admin.py:1445 -#, fuzzy, python-format +#: cps/admin.py:1457 +#, python-format msgid "Success! Password for user %(user)s reset" -msgstr "사용자 %(user)s의 비밀번호 재설정" +msgstr "사용자 %(user)s의 비밀번호가 재설정되었습니다." -#: cps/admin.py:1451 -#, fuzzy +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." -msgstr "먼저 SMTP 메일 설정을 구성하십시오..." +msgstr "SMTP를 설정해주세요." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" -msgstr "로그 파일 뷰어" - -#: cps/admin.py:1528 -msgid "Requesting update package" -msgstr "업데이트 패키지 요청" - -#: cps/admin.py:1529 -msgid "Downloading update package" -msgstr "업데이트 패키지 다운로드" - -#: cps/admin.py:1530 -msgid "Unzipping update package" -msgstr "업데이트 패키지 압축 풀기" - -#: cps/admin.py:1531 -msgid "Replacing files" -msgstr "파일 교체" - -#: cps/admin.py:1532 -msgid "Database connections are closed" -msgstr "데이터베이스 연결이 닫힙니다" - -#: cps/admin.py:1533 -msgid "Stopping server" -msgstr "서버 중지" - -#: cps/admin.py:1534 -msgid "Update finished, please press okay and reload page" -msgstr "업데이트가 완료되었습니다. 확인을 누르고 페이지를 새로고침하세요" - -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 -msgid "Update failed:" -msgstr "업데이트 실패:" - -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 -msgid "HTTP Error" -msgstr "HTTP 오류" - -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 -msgid "Connection error" -msgstr "연결 오류" - -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 -msgid "Timeout while establishing connection" -msgstr "연결 설정 중 시간 초과" - -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 -msgid "General error" -msgstr "일반 오류" - -#: cps/admin.py:1539 -msgid "Update file could not be saved in temp dir" -msgstr "임시 디렉토리에 업데이트 파일을 저장할 수 없습니다" +msgstr "로그 파일 보기" #: cps/admin.py:1540 +msgid "Requesting update package" +msgstr "업데이트 패키지 요청 중" + +#: cps/admin.py:1541 +msgid "Downloading update package" +msgstr "업데이트 패키지 다운로드 중" + +#: cps/admin.py:1542 +msgid "Unzipping update package" +msgstr "업데이트 패키지 압축 해제 중" + +#: cps/admin.py:1543 +msgid "Replacing files" +msgstr "파일 적용 중" + +#: cps/admin.py:1544 +msgid "Database connections are closed" +msgstr "데이터베이스 연결이 종료됩니다." + +#: cps/admin.py:1545 +msgid "Stopping server" +msgstr "서버를 중지하고 있습니다." + +#: cps/admin.py:1546 +msgid "Update finished, please press okay and reload page" +msgstr "업데이트가 완료되었습니다. 확인을 누르고 페이지를 새로고침해주세요." + +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 +msgid "Update failed:" +msgstr "업데이트 실패: " + +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +msgid "HTTP Error" +msgstr "HTTP 오류가 발생했습니다." + +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 +msgid "Connection error" +msgstr "연결 오류가 발생했습니다." + +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 +msgid "Timeout while establishing connection" +msgstr "연결 시도 중 시간이 초과되었습니다." + +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 +msgid "General error" +msgstr "일반 오류가 발생했습니다." + +#: cps/admin.py:1551 +msgid "Update file could not be saved in temp dir" +msgstr "임시 폴더에 업데이트 파일을 저장할 수 없습니다." + +#: cps/admin.py:1552 msgid "Files could not be replaced during update" -msgstr "업데이트하는 동안 파일을 교체할 수 없습니다" +msgstr "업데이트 중 파일을 변경할 수 없습니다." -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" -msgstr "하나 이상의 LDAP 사용자를 추출하지 못했습니다" +msgstr "최소 한 명의 LDAP 사용자 정보를 가져오지 못했습니다." -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" -msgstr "하나 이상의 LDAP 사용자를 생성하지 못했습니다" +msgstr "최소 한 명의 LDAP 사용자 정보를 생성하지 못했습니다." -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "오류: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" -msgstr "오류: LDAP 서버의 응답으로 사용자가 반환되지 않았습니다" +msgstr "오류: LDAP 서버에서 반환된 사용자가 없습니다." -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" -msgstr "데이터베이스에서 하나 이상의 LDAP 사용자를 찾을 수 없습니다" +msgstr "데이터베이스에서 찾을 수 없는 LDAP 사용자가 있습니다." -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" -msgstr "{} 사용자 가져오기 성공" +msgstr "{} 명의 사용자를 가져왔습니다." -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "책 경로가 유효하지 않습니다." + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" -msgstr "올바르지 않은 DB 위치. 올바른 경로 입력 필요" +msgstr "데이터베이스 경로가 유효하지 않습니다." -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" -msgstr "쓰기 권한이 없는 DB" +msgstr "데이터베이스에 쓰기 권한이 없습니다." -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" -msgstr "올바르지 않은 키 파일 위치. 올바른 경로 입력 필요" +msgstr "키 파일 경로가 유효하지 않습니다." -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" -msgstr "올바르지 않은 인증서 파일 위치. 올바른 경로 입력 필요" +msgstr "인증서 파일 경로가 유효하지 않습니다." -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" -msgstr "비밀번호 길이는 1에서 40 사이여야 합니다." +msgstr "비밀번호는 1~40자 사이로 입력해주세요." -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" -msgstr "데이터베이스 설정이 업데이트 되었습니다" +msgstr "데이터베이스 설정이 업데이트되었습니다." -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" -msgstr "데이터베이스 구성" +msgstr "데이터베이스 설정" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." -msgstr "모든 필드를 채워주십시오!" - -#: cps/admin.py:1926 -msgid "E-mail is not from valid domain" -msgstr "유효한 도메인에서 온 이메일이 아니" - -#: cps/admin.py:1932 -msgid "Add new user" -msgstr "새 사용자 추가" - -#: cps/admin.py:1943 -#, python-format -msgid "User '%(user)s' created" -msgstr "사용자 '%(user)s'이(가) 생성됨" +msgstr "모든 항목을 입력해주세요." #: cps/admin.py:1949 -msgid "Oops! An account already exists for this Email. or name." -msgstr "동일한 이메일 주소 또는 이름이 이미 등록되어 있습니다." +msgid "E-mail is not from valid domain" +msgstr "이메일 주소가 올바르지 않습니다." -#: cps/admin.py:1979 +#: cps/admin.py:1955 +msgid "Add new user" +msgstr "새 사용자 생성" + +#: cps/admin.py:1966 +#, python-format +msgid "User '%(user)s' created" +msgstr "사용자 '%(user)s'이(가) 생성되었습니다." + +#: cps/admin.py:1972 +msgid "Oops! An account already exists for this Email. or name." +msgstr "이미 등록된 이메일 주소 또는 이름입니다." + +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" -msgstr "사용자 '%(nick)s'이(가) 삭제됨" +msgstr "사용자 '%(nick)s'이(가) 삭제되었습니다." -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" -msgstr "게스트 사용자는 삭제할 수 없습니다" +msgstr "게스트 사용자는 삭제할 수 없습니다." -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" -msgstr "관리자 계정이 하나 뿐일 때는 관리자 권한을 삭제할 수 없음" +msgstr "남은 관리자가 없어 사용자를 삭제할 수 없습니다." -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" -msgstr "이메일은 반드시 입력해야 하며 유효한 이메일이어야 합니다." +msgstr "이메일 주소는 비워둘 수 없고, 올바른 형식이어야 합니다." -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" -msgstr "사용자 '%(nick)s'가 업데이트 됨" +msgstr "사용자 '%(nick)s의 정보를 수정했습니다." + +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "검색" #: cps/converter.py:31 msgid "not installed" -msgstr "설치되지 않음" +msgstr "설치되지 않았습니다." #: cps/converter.py:32 msgid "Execution permissions missing" -msgstr "실행 권한 누락" +msgstr "실행 권한이 없습니다." -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "사용자 정의 열 번호 %(column)d이(가) calibre 데이터베이스에 없습니다" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "None" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "선택한 책 제목을 사용할 수 없습니다. 파일이 존재하지 않거나 액세스할 수 없습니다" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "표지를 업로드 할 수 있는 권한이 없는 사용자입니다." - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "식별자는 대소문자를 구분하지 않으며 이전 식별자를 덮어씁니다" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "메타데이터가 성공적으로 업데이트되었습니다" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "책 편집 중 오류 발생: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" -msgstr "파일 %(file)s 업로드됨" +msgstr "파일 %(file)s이(가) 업로드되었습니다." -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" -msgstr "변환을 위한 소스 또는 대상 형식이 누락되었습니다" +msgstr "변환할 원본 또는 대상 형식이 지정되지 않았습니다." -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" -msgstr "책이 %(book_format)s(으)로 변환하기 위해 대기 중입니다" +msgstr "%(book_format)s 형식으로 변환하기 위해 작업에 추가되었습니다." -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" -msgstr "이 책을 변환하는 동안 오류가 발생했습니다: %(res)s" +msgstr "책 변환 중 오류가 발생했습니다: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "업로드한 책이 라이브러리에 있을 수 있음. 새로 업로드하기 전에 확인 필요: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "선택한 책을 사용할 수 없습니다. 파일이 없거나 접근할 수 없습니다." -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "표지 업로드 권한이 없습니다." + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "식별자는 대소문자 구분 없어 덮어씁니다." + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" -msgstr "'%(langname)s'은(는) 유효한 언어가 아닙니다" +msgstr "'%(langname)s'은(는) 올바른 언어가 아닙니다." -#: cps/editbooks.py:756 cps/editbooks.py:1192 -#, fuzzy +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "메타데이터가 업데이트되었습니다." + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "책 편집 중 오류가 발생했습니다: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "업로드된 책이 이미 존재하는 것 같습니다. 확인이 필요합니다: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" -msgstr "파일 확장자 '%(ext)s'은(는) 이 서버에 업로드할 수 없습니다" +msgstr "업로드할 수 없는 파일 형식입니다." -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" -msgstr "파일 확장자 '%(ext)s'은(는) 이 서버에 업로드할 수 없습니다" +msgstr "파일 확장자 '%(ext)s'은(는) 서버에 업로드할 수 없습니다" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" -msgstr "업로드할 파일에는 확장자가 있어야 합니다" +msgstr "확장자가 없는 파일은 업로드할 수 없습니다." -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" -msgstr "파일 %(filename)s을(를) 임시 디렉토리에 저장할 수 없습니다" +msgstr "파일 %(filename)s을(를) 임시 폴더에 저장할 수 없습니다." -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" -msgstr "표지 파일%(file)s를 이동하지 못했습니다.:%(error)s" +msgstr "표지 파일 %(file)s을(를) 이동하지 못했습니다 :%(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" -msgstr "책 형식이 성공적으로 삭제되었습니다" +msgstr "책 형식이 삭제되었습니다." -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" -msgstr "책이 성공적으로 삭제되었습니다" +msgstr "책이 삭제되었습니다." -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" -msgstr "책을 삭제할 수 있는 권한이 없습니다." +msgstr "책을 삭제할 권한이 없습니다." -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "메타데이터 편집" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" -msgstr "%(seriesindex)s은(는) 유효한 숫자가 아닙니다. 건너뜁니다" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "시리즈 순서 %(seriesindex)s은(는) 올바르지 않아 건너뜁니다." -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" -msgstr "추가 파일 유형을 업로드 할 권한이 없는 사용자입니다." +msgstr "추가 파일 형식을 업로드할 권한이 없습니다." -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." -msgstr "%(path)s 경로를 생성하지 못했습니다(권한이 없음)." +msgstr "쓰기 권한이 없어 %(path)s 경로 생성에 실패했습니다." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." -msgstr "%(file)s 파일을 저장하지 못했습니다." +msgstr "%(file)s 파일 저장에 실패했습니다." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "파일 형식 %(ext)s이(가) %(book)s에 추가되었습니다" #: cps/gdrive.py:58 msgid "Google Drive setup not completed, try to deactivate and activate Google Drive again" -msgstr "Google 드라이브 설정이 완료되지 않았습니다. Google 드라이브를 비활성화했다가 다시 활성화해 보세요" +msgstr "Google 드라이브 설정이 완료되지 않았습니다. Google 드라이브를 비활성화한 후 다시 활성화해 주세요." #: cps/gdrive.py:96 msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" -msgstr "콜백 도메인이 확인되지 않았습니다. 단계에 따라 Google 개발자 콘솔에서 도메인을 확인하세요" +msgstr "콜백 도메인이 인증되지 않았습니다. Google 개발자 콘솔에서 도메인 인증 절차를 진행해 주세요." -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" -msgstr "책 ID에 대한 %(format)s 형식을 찾을 수 없음: %(book)d" +msgstr "책 ID %(book)d의 %(format)s 형식을 찾을 수 없습니다." -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" -msgstr "%(format)s을(를) Google 드라이브에서 찾을 수 없음: %(fn)s" +msgstr "Google 드라이브에서 %(format)s 파일을 찾을 수 없습니다: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" -msgstr "%(format)s을(를) 찾을 수 없음: %(fn)s" +msgstr "%(format)s을(를) 찾을 수 없습니다: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 -#, fuzzy +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" -msgstr "킨들로 보내기" +msgstr "전자책 리더로 보내기" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 -#, fuzzy +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." -msgstr "이 이메일은 Calibre Web을 통해 전송되었습니다." +msgstr "이 이메일은 Calibre-Web으로 전송되었습니다." -#: cps/helper.py:120 -#, fuzzy +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "Calibre-Web 테스트 이메일" -#: cps/helper.py:121 -#, fuzzy +#: cps/helper.py:124 msgid "Test Email" msgstr "테스트 이메일" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Calibre-Web 시작하기" -#: cps/helper.py:143 -#, fuzzy, python-format +#: cps/helper.py:146 +#, python-format msgid "Registration Email for user: %(name)s" -msgstr "사용자 등록 이메일: %(name)s" +msgstr "사용자 %(name)s의 등록 이메일" -#: cps/helper.py:154 cps/helper.py:160 -#, fuzzy, python-format +#: cps/helper.py:157 cps/helper.py:163 +#, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" -msgstr "%(orig)s를 %(format)s로 변환하고 킨들로 보내기" +msgstr "%(orig)s을(를) %(format)s로 변환하여 전자책 리더로 보내기" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 -#, fuzzy, python-format +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 +#, python-format msgid "Send %(format)s to eReader" -msgstr "킨들에 %(format)s 보내기" +msgstr "%(format)s을 전자책 리더로 보내기" -#: cps/helper.py:227 -#, fuzzy, python-format +#: cps/helper.py:230 +#, python-format msgid "%(book)s send to eReader" -msgstr "%(book)s을 킨들로 보내기" +msgstr "%(book)s을(를) 전자책 리더로 보내기" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" -msgstr "요청한 파일을 읽을 수 없습니다. 올바른 권한인가요?" +msgstr "요청한 파일을 읽을 수 없습니다. 파일 접근 권한을 확인해주세요." -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" -msgstr "읽기 상태를 설정할 수 없음: {}" +msgstr "읽기 상태를 설정할 수 없습니다: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" -msgstr "책 %(id)s에 대한 책 폴더를 삭제하지 못했습니다. 경로에 하위 폴더가 있습니다: %(path)s" +msgstr "책 %(id)s의 폴더 삭제에 실패했습니다. %(path)s에 하위 폴더가 있습니다." -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" -msgstr "%(id)s 도서 삭제 실패: %(message)s" +msgstr "책 %(id)s 삭제에 실패했습니다: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" -msgstr "데이터베이스에서만 책 %(id)s 을(를) 삭제 중, 데이터베이스의 책 경로가 유효하지 않음: %(path)s" +msgstr "데이터베이스 경로가 올바르지 않아 책 %(id)s을(를) 데이터베이스에서만 삭제합니다: %(path)s" -#: cps/helper.py:463 -#, fuzzy, python-format +#: cps/helper.py:439 +#, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "제목 이름을 '%(src)s'에서 '%(dest)s'(으)로 변경하지 못했습니다. 오류: %(error)s" +msgstr "저자명 '%(src)s'에서 '%(dest)s'(으)로 변경하지 못했습니다: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" -msgstr "Google 드라이브에서 %(file)s 파일을 찾을 수 없습니다" +msgstr "Google 드라이브에서 %(file)s 파일을 찾을 수 없습니다." -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" -msgstr "제목 이름을 '%(src)s'에서 '%(dest)s'(으)로 변경하지 못했습니다. 오류: %(error)s" +msgstr "제목을 '%(src)s'에서 '%(dest)s'(으)로 변경하지 못했습니다: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" -msgstr "Google 드라이브에서 책 경로 %(path)s을(를) 찾을 수 없습니다" +msgstr "Google 드라이브에서 경로 %(path)s을(를) 찾을 수 없습니다." -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" -msgstr "다른 계정에서 사용하고 있는 이메일 주소입니다." +msgstr "이미 사용 중인 이메일 주소입니다." -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" -msgstr "등록되어 있는 username입니다" +msgstr "이미 사용 중인 사용자 이름입니다." -#: cps/helper.py:702 -#, fuzzy +#: cps/helper.py:679 msgid "Invalid Email address format" -msgstr "이메일 주소 형식이 잘못되었습니다" +msgstr "올바른 이메일 주소가 아닙니다." -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" -msgstr "규칙에 어긋나는 비밀번호입니다." +msgstr "비밀번호가 설정된 규칙에 맞지 않습니다." -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "표지 업로드에 필요한 Python 모듈 'advocate'이 설치되지 않았습니다." -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" -msgstr "표지 다운로드 중 오류 발생" +msgstr "표지 다운로드 중 오류가 발생했습니다." -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" -msgstr "표지 형식 오류" +msgstr "표지 형식 오류입니다." -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" -msgstr "표지 업로드를 위해 localhost 또는 로컬 네트워크에 액세스할 수 없습니다." +msgstr "표지 업로드를 위해 로컬 호스트나 로컬 네트워크에 접근할 수 없습니다." -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" -msgstr "표지 경로 생성 실패" +msgstr "표지 경로 생성에 실패했습니다." + +#: cps/helper.py:889 +msgid "Cover-file is not a valid image file, or could not be stored" +msgstr "표지 파일이 올바른 이미지 파일이 아니거나 저장할 수 없습니다." + +#: cps/helper.py:900 +msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" +msgstr "jpg/jpeg/png/webp/bmp 형식만 지원합니다." #: cps/helper.py:912 -msgid "Cover-file is not a valid image file, or could not be stored" -msgstr "표지 파일이 유효한 이미지 파일이 아니거나 저장할 수 없습니다" - -#: cps/helper.py:923 -msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" -msgstr "표지는 jpg/jpeg/png/webp/bmp 파일만 지원됩니다" - -#: cps/helper.py:935 msgid "Invalid cover file content" -msgstr "잘못된 표지 파일 콘텐츠" +msgstr "올바르지 않은 표지 파일입니다." -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" -msgstr "표지 파일로 jpg/jpeg 파일만 지원됩니다" +msgstr "표지 파일은 jpg/jpeg 형식만 지원합니다." -#: cps/helper.py:1011 cps/helper.py:1168 -#, fuzzy +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" -msgstr "발견" +msgstr "표지" + +#: cps/helper.py:1006 +msgid "UnRar binary file not found" +msgstr "UnRar 바이너리 파일을 찾을 수 없습니다." + +#: cps/helper.py:1017 +msgid "Error executing UnRar" +msgstr "UnRar 실행 중 오류가 발생했습니다." + +#: cps/helper.py:1025 +msgid "Could not find the specified directory" +msgstr "지정된 디렉토리를 찾을 수 없습니다." #: cps/helper.py:1028 -msgid "UnRar binary file not found" -msgstr "UnRar 바이너리 파일을 찾을 수 없습니다" - -#: cps/helper.py:1039 -#, fuzzy -msgid "Error executing UnRar" -msgstr "UnRar 실행 오류" - -#: cps/helper.py:1047 -msgid "Could not find the specified directory" -msgstr "" - -#: cps/helper.py:1050 msgid "Please specify a directory, not a file" -msgstr "" +msgstr "파일이 아닌 폴더를 지정하세요." -#: cps/helper.py:1064 -#, fuzzy +#: cps/helper.py:1042 msgid "Calibre binaries not viable" -msgstr "쓰기 권한이 없는 DB" +msgstr "Calibre 바이너리 파일을 사용할 수 없습니다." -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" -msgstr "" +msgstr "Calibre 바이너리 파일이 누락되었습니다: %(missing)s" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" -msgstr "실행 권한 누락" +msgstr "실행 권한이 누락되었습니다: %(missing)s" -#: cps/helper.py:1080 -#, fuzzy +#: cps/helper.py:1058 msgid "Error executing Calibre" -msgstr "UnRar 실행 오류" +msgstr "Calibre 실행 오류" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" -msgstr "메타데이터 백업을 위해 모든 도서를 대기열에 추가" +msgstr "메타데이터 백업이 작업에 추가되었습니다." -#: cps/kobo_auth.py:90 -#, fuzzy +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" -msgstr "코보(kobo) 장치에 대한 유효한 api_endpoint를 얻으려면 로컬 호스트가 아닌 곳에서 calibre-web에 액세스하십시오" +msgstr "Kobo 장치에 대한 유효한 api_endpoint를 얻으려면 로컬 호스트가 아닌 곳에서 Calibre-Web에 접근하세요." -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo 설정" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "%(provider)s에 등록" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" -msgstr "다음 사용자로 로그인했습니다: '%(nickname)s'" +msgstr "%(nickname)s로 로그인했습니다." -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" -msgstr "%(oauth)s에 성공적으로 연결됨" +msgstr "%(oauth)s에 성공적으로 연결되었습니다." -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" -msgstr "로그인 실패, OAuth 계정과 연결된 사용자 없음" +msgstr "로그인에 실패했습니다. OAuth 계정과 연결된 사용자가 없습니다." -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" -msgstr "%(oauth)s에 연결 해제 성공" +msgstr "%(oauth)s에 연결 해제에 성공했습니다." -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" -msgstr "%(oauth)s에 연결 해제 실패" +msgstr "%(oauth)s에 연결 해제에 실패했습니다." -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" -msgstr "%(oauth)s에 연결되지 않음" +msgstr "%(oauth)s에 연결되지 않았습니다." -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." -msgstr "GitHub에 로그인하지 못했습니다." +msgstr "GitHub 로그인에 실패했습니다." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." -msgstr "GitHub에서 사용자 정보를 가져오지 못했습니다." +msgstr "GitHub에서 사용자 정보를 불러오지 못했습니다." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." -msgstr "구글 로그인에 실패했습니다." +msgstr "Google 로그인에 실패했습니다." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." -msgstr "Google에서 사용자 정보를 가져오지 못했습니다." +msgstr "Google에서 사용자 정보를 불러오지 못했습니다." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." -msgstr "GitHub Oauth 오류입니다. 나중에 다시 시도하세요." +msgstr "GitHub Oauth 오류입니다. 잠시 후 다시 시도해 주세요." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" -msgstr "GitHub 인증 오류: {}" +msgstr "GitHub 인증 오류가 발생했습니다: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." -msgstr "Google 인증 오류입니다. 나중에 다시 시도하세요." +msgstr "Google 인증 오류입니다. 잠시 후 다시 시도해 주세요." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" -msgstr "Google 인증 오류: {}" +msgstr "Google 인증 오류가 발생했습니다: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Stars" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "로그인" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" -msgstr "토큰을 찾을 수 없습니다" +msgstr "토큰을 찾을 수 없습니다." -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" -msgstr "토큰이 만료되었습니다" +msgstr "토큰이 만료되었습니다." -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" -msgstr "성공! 기기로 돌아가주세요" +msgstr "인증되었습니다. 기기로 돌아가 주세요." -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" -msgstr "책" +msgstr "전체" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "최근 책 보기" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" -msgstr "인기있는 책" +msgstr "인기" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" -msgstr "주목받는 책 보기" +msgstr "인기 책 보기" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" -msgstr "다운로드된 책" +msgstr "받은 적 있는" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" -msgstr "다운로드된 책 보기" +msgstr "다운로드 받은 적 있는 책 보기" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" -msgstr "평점이 높은 책" +msgstr "베스트" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" -msgstr "평점이 높은 책 보기" +msgstr "평점 높은 책 보기" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" -msgstr "읽은 책" +msgstr "읽음" -#: cps/render_template.py:64 -#, fuzzy +#: cps/render_template.py:63 msgid "Show Read and Unread" -msgstr "읽은 책과 읽지 않은 책 보기" +msgstr "읽은 책과 안 읽은 책 보기" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" -msgstr "읽지 않은 책" +msgstr "읽지 않음" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "읽지 않은 책 보기" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" -msgstr "발견" +msgstr "둘러보기" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" -msgstr "무작위 추천" +msgstr "무작위 추천 보기" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "카테고리" -#: cps/render_template.py:74 cps/templates/user_table.html:158 -#, fuzzy +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" -msgstr "카테고리별 보기" +msgstr "카테고리 보기" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "시리즈" -#: cps/render_template.py:77 cps/templates/user_table.html:157 -#, fuzzy +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" -msgstr "시리즈별 보기" +msgstr "시리즈 보기" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "저자" -#: cps/render_template.py:80 cps/templates/user_table.html:160 -#, fuzzy +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" -msgstr "저자별 보기" +msgstr "저자 보기" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "출판사" -#: cps/render_template.py:84 cps/templates/user_table.html:163 -#, fuzzy +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" -msgstr "출판사별 보기" +msgstr "출판사 보기" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "언어" -#: cps/render_template.py:88 cps/templates/user_table.html:155 -#, fuzzy +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" -msgstr "언어별 보기" +msgstr "언어 보기" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "평점" -#: cps/render_template.py:91 cps/templates/user_table.html:164 -#, fuzzy +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" -msgstr "평점별 보기" +msgstr "평점 보기" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" -msgstr "파일 유형" +msgstr "파일 형식" -#: cps/render_template.py:94 cps/templates/user_table.html:165 -#, fuzzy +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" -msgstr "파일 유형별 보기" +msgstr "파일 형식 보기" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" -msgstr "보관된 책" +msgstr "보관함" -#: cps/render_template.py:98 cps/templates/user_table.html:166 -#, fuzzy +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "보관된 책 보기" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "책 목록" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "책 목록 보기" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "검색" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " -msgstr "Published after " +msgstr "발행일 이후" -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " -msgstr "Published before " +msgstr "발행일 이전" -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "평점 <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "평점 >= %(rating)s" -#: cps/search.py:221 -#, fuzzy, python-format +#: cps/search.py:234 +#, python-format msgid "Read Status = '%(status)s'" msgstr "읽은 상태 = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" -msgstr "사용자 정의 열을 검색하는 동안 오류가 발생했습니다. Calibre-Web을 다시 시작하십시오" +msgstr "사용자 정의 열을 검색하는 동안 오류가 발생했습니다. Calibre-Web을 다시 시작해 주세요." -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "상세 검색" #: cps/shelf.py:49 cps/shelf.py:111 msgid "Invalid shelf specified" -msgstr "잘못된 책장 지정" +msgstr "잘못된 서재가 지정되었습니다." #: cps/shelf.py:55 msgid "Sorry you are not allowed to add a book to that shelf" -msgstr "죄송합니다. 해당 책장에 책을 추가할 권한이 없습니다" +msgstr "추가할 권한이 없습니다." #: cps/shelf.py:64 #, python-format msgid "Book is already part of the shelf: %(shelfname)s" -msgstr "책이 이미 책장에 등록됨: %(shelfname)s" +msgstr "이미 등록된 책입니다: %(shelfname)s" #: cps/shelf.py:77 #, python-format msgid "%(book_id)s is a invalid Book Id. Could not be added to Shelf" -msgstr "" +msgstr "추가할 수 없습니다. %(book_id)s는 잘못된 책 ID입니다." #: cps/shelf.py:97 #, python-format msgid "Book has been added to shelf: %(sname)s" -msgstr "책장에 책이 등록됨: %(sname)s" +msgstr "책이 등록되었습니다: %(sname)s" #: cps/shelf.py:116 msgid "You are not allowed to add a book to the shelf" -msgstr "책장에 책을 추가할 권한이 없습니다" +msgstr "책을 추가할 권한이 없습니다." #: cps/shelf.py:134 #, python-format msgid "Books are already part of the shelf: %(name)s" -msgstr "책장에 이미 책이 등록됨: %(name)s" +msgstr "이미 등록된 책입니다: %(name)s" #: cps/shelf.py:146 #, python-format msgid "Books have been added to shelf: %(sname)s" -msgstr "책이 책장에 등록됨: %(sname)s" +msgstr "책들이 등록되었습니다: %(sname)s" #: cps/shelf.py:153 #, python-format msgid "Could not add books to shelf: %(sname)s" -msgstr "책장에 책을 추가할 수 없음: %(sname)s" +msgstr "책을 추가할 수 없습니다: %(sname)s" #: cps/shelf.py:199 #, python-format msgid "Book has been removed from shelf: %(sname)s" -msgstr "책이 책장에서 삭제됨: %(sname)s" +msgstr "책이 서재에서 삭제되었습니다: %(sname)s" #: cps/shelf.py:208 msgid "Sorry you are not allowed to remove a book from this shelf" -msgstr "죄송합니다. 이 서가에서 책을 삭제할 권한이 없습니다" +msgstr "서재에서 책을 삭제할 권한이 없습니다." -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" -msgstr "책장 추가" +msgstr "서재 생성" #: cps/shelf.py:226 msgid "Sorry you are not allowed to edit this shelf" -msgstr "죄송합니다. 이 책장을 편집할 권한이 없습니다" +msgstr "서재를 수정할 권한이 없습니다." #: cps/shelf.py:228 msgid "Edit a shelf" -msgstr "책장 편집" +msgstr "서재 편집" #: cps/shelf.py:237 msgid "Error deleting Shelf" -msgstr "서제를 삭제하는 동안 오류 발생" +msgstr "서재를 삭제하는 중 오류가 발생했습니다." #: cps/shelf.py:239 -#, fuzzy msgid "Shelf successfully deleted" -msgstr "책이 성공적으로 삭제되었습니다" +msgstr "서재가 삭제되었습니다." #: cps/shelf.py:289 #, python-format msgid "Change order of Shelf: '%(name)s'" -msgstr "책장 순서 변경: '%(name)s'" +msgstr "서재 순서가 변경되었습니다: '%(name)s'" #: cps/shelf.py:324 msgid "Sorry you are not allowed to create a public shelf" -msgstr "죄송합니다. 이 공개 책장을 만들 권한이 없습니다" +msgstr "공개 서재를 생성할 권한이 없습니다." #: cps/shelf.py:341 #, python-format msgid "Shelf %(title)s created" -msgstr "%(title)s 책장이 추가됨" +msgstr "서재 %(title)s가 생성되었습니다." #: cps/shelf.py:344 #, python-format msgid "Shelf %(title)s changed" -msgstr "%(title)s 책장이 변경됨" +msgstr "서재 %(title)s가 변경되었습니다." #: cps/shelf.py:358 msgid "There was an error" -msgstr "오류가 발생하였습니다" +msgstr "오류가 발생했습니다." #: cps/shelf.py:380 #, python-format msgid "A public shelf with the name '%(title)s' already exists." -msgstr "이름이 '%(title)s'인 공개 책장이 이미 있습니다." +msgstr "'%(title)s' 공개 서재가 이미 존재합니다." #: cps/shelf.py:391 #, python-format msgid "A private shelf with the name '%(title)s' already exists." -msgstr "이름이 '%(title)s'인 개인 책장이 이미 있습니다." +msgstr "'%(title)s' 개인 서재가 이미 존재합니다." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" -msgstr "책장: '%(name)s'" +msgstr "서재: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" -msgstr "책장을 여는 동안 오류가 발생했습니다. 책장이 존재하지 않거나 접근할 수 없습니다" +msgstr "서재를 여는 동안 오류가 발생했습니다. 서재가 존재하지 않거나 접근할 수 없습니다" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "작업" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" -msgstr "대기중" +msgstr "대기 중" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "실패" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "시작됨" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" -msgstr "종료" +msgstr "종료됨" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "종료됨" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "취소됨" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "알 수 없는 상태" #: cps/updater.py:433 cps/updater.py:444 cps/updater.py:545 cps/updater.py:560 msgid "Unexpected data while reading update information" -msgstr "업데이트 정보를 읽는 동안 예기치 않은 데이터" +msgstr "업데이트 정보를 읽는 중 알 수 없는 문제가 발생했습니다." #: cps/updater.py:440 cps/updater.py:552 msgid "No update available. You already have the latest version installed" -msgstr "업데이트가 없습니다. 이미 최신 버전이 설치되어 있습니다" +msgstr "업데이트가 없습니다. 최신 버전이 설치되어 있습니다." #: cps/updater.py:458 msgid "A new update is available. Click on the button below to update to the latest version." -msgstr "새로운 업데이트가 있습니다. 아래 버튼을 클릭하여 최신 버전으로 업데이트하십시오." +msgstr "새로운 업데이트가 있습니다. 아래 버튼을 클릭해 최신 버전으로 업데이트하세요." #: cps/updater.py:476 msgid "Could not fetch update information" -msgstr "업데이트 정보를 가져올 수 없습니다" +msgstr "업데이트 정보를 가져올 수 없습니다." #: cps/updater.py:486 msgid "Click on the button below to update to the latest stable version." -msgstr "새로운 업데이트가 있습니다. 아래 버튼을 클릭하여 최신 버전으로 업데이트하십시오." +msgstr "아래 버튼을 클릭해 최신 버전으로 업데이트하세요." #: cps/updater.py:495 cps/updater.py:509 cps/updater.py:520 #, python-format msgid "A new update is available. Click on the button below to update to version: %(version)s" -msgstr "최신 안정 버전 %(version)s으로 업데이트하려면 아래 버튼을 클릭하세요" +msgstr "새로운 업데이트가 있습니다. %(version)s으로 업데이트하려면 아래 버튼을 클릭하세요." #: cps/updater.py:538 msgid "No release information available" -msgstr "출시 정보가 없습니다" +msgstr "업데이트 정보가 없습니다." -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" -msgstr "발견(무작위 도서)" +msgstr "추천 (무작위)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" -msgstr "인기있는 책(가장 많이 다운로드됨)" +msgstr "인기 책 (가장 많이 다운로드된 책)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "%(user)s이(가) 다운로드한 책" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "저자: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "출판사: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "시리즈: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" -msgstr "평가: 없음음" +msgstr "평점: 없음" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" -msgstr "평가: %(rating)s 별" +msgstr "평점: %(rating)s" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" -msgstr "파일 유형: %(format)s" +msgstr "파일 형식: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "카테고리: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "언어: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "다운로드" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "평점 목록" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" -msgstr "파일 유형 목록" - -#: cps/web.py:1249 -#, fuzzy -msgid "Please configure the SMTP mail settings first..." -msgstr "먼저 SMTP 메일 설정을 구성하십시오..." - -#: cps/web.py:1256 -#, python-format -msgid "Success! Book queued for sending to %(eReadermail)s" -msgstr "성공적으로 %(eReadermail)s에 보내기 예약이 되었습니다" +msgstr "파일 형식 목록" #: cps/web.py:1259 +msgid "Please configure the SMTP mail settings first..." +msgstr "SMTP 메일 설정을 해주세요." + +#: cps/web.py:1265 +#, python-format +msgid "Success! Book queued for sending to %(eReadermail)s" +msgstr "%(eReadermail)s에 전송 작업이 추가되었습니다." + +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" -msgstr "책을 보내는 중에 오류 발생: %(res)s" +msgstr "전송 중 오류가 발생했습니다: %(res)s" -#: cps/web.py:1261 -#, fuzzy +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." -msgstr "Kindle로 보내는 유효한 이메일 주소로 프로필을 업데이트하십시오." +msgstr "문제가 발생했습니다. 프로필에 유효한 전자책 리더 이메일을 설정해 주세요." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" -msgstr "1분 이상 지난 후 다음 사용자를 등록하세요." +msgstr "다음 사용자를 등록은 1분 뒤 가능합니다." -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "등록" -#: cps/web.py:1281 cps/web.py:1385 -#, fuzzy +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" -msgstr "이메일 서버가 구성되지 않았습니다. 관리자에게 문의하십시오!" +msgstr "Flask-Limiter 시스템 연결에 문제가 발생했습니다. 관리자에게 문의하세요." -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." -msgstr "이메일 서버가 구성되지 않았습니다. 관리자에게 문의하십시오!" +msgstr "이메일 서버가 설정되지 않았습니다. 관리자에게 문의하세요." -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." -msgstr "이메일을 등록할 수 없습니다" +msgstr "사용할 수 없는 이메일입니다." -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." -msgstr "확인을 위한 이메일이 발송되었습니다." +msgstr "인증 이메일을 발송했습니다." -#: cps/web.py:1368 cps/web.py:1391 -#, fuzzy +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" -msgstr "LDAP 인증을 활성화할 수 없습니다" +msgstr "LDAP 인증을 활성화할 수 없습니다." -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" -msgstr "1분 이상 지난 후 로그인을 하세요." +msgstr "1분 후 시도해 주세요." -#: cps/web.py:1400 -#, fuzzy, python-format +#: cps/web.py:1408 +#, python-format msgid "you are now logged in as: '%(nickname)s'" -msgstr "다음 사용자로 로그인했습니다: '%(nickname)s'" +msgstr "'%(nickname)s'로 로그인했습니다." -#: cps/web.py:1407 -#, fuzzy, python-format +#: cps/web.py:1415 +#, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" -msgstr "대체 로그인: '%(nickname)s', LDAP 서버에 연결할 수 없음 또는 사용자를 알 수 없음" +msgstr "'%(nickname)s'(으)로 대체 로그인했습니다. LDAP 서버에 연결할 수 없거나 존재하지 않는 사용자입니다." -#: cps/web.py:1412 -#, fuzzy, python-format +#: cps/web.py:1420 +#, python-format msgid "Could not login: %(message)s" -msgstr "로그인 실패: %(message)s" +msgstr "로그인할 수 없습니다: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 -#, fuzzy +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" -msgstr "잘못된 사용자명 또는 비밀번호" +msgstr "사용자 이름 또는 비밀번호가 올바르지 않습니다." -#: cps/web.py:1423 -#, fuzzy +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "새 비밀번호가 이메일로 전송되었습니다" -#: cps/web.py:1427 -#, fuzzy +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." -msgstr "알 수없는 오류가 발생했습니다. 나중에 다시 시도 해주십시오." - -#: cps/web.py:1429 -#, fuzzy -msgid "Please enter valid username to reset password" -msgstr "비밀번호를 재설정하려면 유효한 사용자 이름을 입력하십시오" +msgstr "알 수 없는 오류가 발생했습니다. 나중에 다시 시도해 주세요." #: cps/web.py:1437 -#, fuzzy, python-format -msgid "You are now logged in as: '%(nickname)s'" -msgstr "다음 사용자로 로그인했습니다: '%(nickname)s'" +msgid "Please enter valid username to reset password" +msgstr "비밀번호를 재설정하려면 올바른 사용자 이름을 입력하세요." -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1445 +#, python-format +msgid "You are now logged in as: '%(nickname)s'" +msgstr "'%(nickname)s'로 로그인했습니다." + +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s 프로필" -#: cps/web.py:1511 -#, fuzzy +#: cps/web.py:1526 msgid "Success! Profile Updated" -msgstr "프로필이 업데이트 됨" +msgstr "프로필이 업데이트되었습니다." -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." -msgstr "등록되어 있는 이메일 주소입니다" +msgstr "이미 등록된 이메일 주소입니다." #: cps/services/gmail.py:59 msgid "Found no valid gmail.json file with OAuth information" -msgstr "인증 정보가 포함된 유효한 gmail.json 파일을 찾을 수 없습니다" +msgstr "인증 정보가 포함된 유요한 gmail.json을 찾을 수 없습니다." -#: cps/tasks/convert.py:108 -#, fuzzy, python-format +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "임시 폴더 비우기" + +#: cps/tasks/convert.py:112 +#, python-format msgid "%(book)s send to E-Reader" -msgstr "%(book)s을 킨들로 보내기" +msgstr "%(book)s을 전자책 리더로 전송" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" -msgstr "Calibre ebook-convert %(tool)s을(를) 찾을 수 없습니다" +msgstr "Calibre ebook-convert %(tool)s을(를) 찾을 수 없습니다." -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" -msgstr "%(format)s 형식을 찾을 수 없습니다" +msgstr "%(format)s 형식을 찾을 수 없습니다." -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" -msgstr "알 수 없는 오류로 인해 전자책 변환이 실패했습니다" +msgstr "알 수 없는 오류로 변환에 실패했습니다." -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" -msgstr "Kepubify 변환 실패: %(error)s" +msgstr "Kepubify 변환에 실패했습니다: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" -msgstr "변환된 파일을 찾을 수 없거나 %(folder)s 폴더에 하나 이상의 파일이 존재합니다" +msgstr "변환된 파일을 찾을 수 없거나 %(folder)s 폴더에 하나 이상의 파일이 존재합니다." -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "\"전자책 변환 실패: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" -msgstr "오류로 인한 Calibre 실패: %(error)s" +msgstr "Calibre가 다음 오류로 실패했습니다: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "변환에 실패했습니다: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "변환" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" -msgstr "Calibre DB를 다시 연결합니다." +msgstr "Calibre 데이터베이스를 다시 연결합니다." -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" -msgstr "이메일일" +msgstr "이메일" #: cps/tasks/metadata_backup.py:34 -#, fuzzy msgid "Backing up Metadata" -msgstr "메타데이터 편집" - -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" +msgstr "메타데이터를 백업하고 있습니다." #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" -msgstr "%(count)개의 표지 섬네일을 생성하였습니다." +msgstr "%(count)s개의 표지 섬네일을 생성했습니다." -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "표지 섬네일" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" -msgstr "{0} 시리즈 섬네일을 생성하였습니다." +msgstr "{0}개의 시리즈 섬네일을 생성했습니다." -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "표지 섬네일 캐시를 삭제합니다." #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "업로드" @@ -1582,19 +1554,18 @@ msgstr "사용자" #: cps/templates/login.html:10 cps/templates/register.html:9 #: cps/templates/user_edit.html:10 cps/templates/user_table.html:134 msgid "Username" -msgstr "사용자명" +msgstr "사용자 이름" #: cps/templates/admin.html:14 cps/templates/register.html:14 #: cps/templates/user_edit.html:15 cps/templates/user_table.html:135 msgid "Email" msgstr "이메일 주소" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 -#, fuzzy +#: cps/templates/admin.html:15 msgid "Send to eReader Email" -msgstr "킨들로 보내기 이메일 주소" +msgstr "전자책 리더로 보낼 이메일" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "관리자" @@ -1604,8 +1575,8 @@ msgstr "관리자" msgid "Password" msgstr "비밀번호" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "다운로드" @@ -1628,7 +1599,7 @@ msgstr "삭제" #: cps/templates/admin.html:26 msgid "Public Shelf" -msgstr "공개 책장" +msgstr "공개 서재" #: cps/templates/admin.html:55 msgid "Import LDAP Users" @@ -1640,29 +1611,28 @@ msgstr "이메일 서버 설정" #: cps/templates/admin.html:67 cps/templates/email_edit.html:31 msgid "SMTP Hostname" -msgstr "SMTP Hostname" +msgstr "SMTP 서버명" #: cps/templates/admin.html:71 cps/templates/email_edit.html:35 msgid "SMTP Port" -msgstr "SMTP Port" +msgstr "SMTP 포트" #: cps/templates/admin.html:75 cps/templates/email_edit.html:39 msgid "Encryption" -msgstr "암호화" +msgstr "보안 연결" #: cps/templates/admin.html:79 cps/templates/email_edit.html:47 msgid "SMTP Login" -msgstr "SMTP Login" +msgstr "SMTP 로그인" #: cps/templates/admin.html:83 cps/templates/admin.html:94 #: cps/templates/email_edit.html:55 msgid "From Email" -msgstr "From Email" +msgstr "보내는 이메일" #: cps/templates/admin.html:90 -#, fuzzy msgid "Email Service" -msgstr "EMail Service" +msgstr "이메일 서비스" #: cps/templates/admin.html:91 msgid "Gmail via Oauth2" @@ -1670,27 +1640,27 @@ msgstr "Gmail via Oauth2" #: cps/templates/admin.html:106 msgid "Configuration" -msgstr "환경 설정" +msgstr "설정" #: cps/templates/admin.html:109 msgid "Calibre Database Directory" -msgstr "Calibre DB 폴더" +msgstr "Calibre 데이터베이스 경로" #: cps/templates/admin.html:113 cps/templates/config_edit.html:68 msgid "Log Level" -msgstr "Log Level" +msgstr "로그 레벨" #: cps/templates/admin.html:117 msgid "Port" -msgstr "Port" +msgstr "포트" #: cps/templates/admin.html:122 msgid "External Port" -msgstr "External Port" +msgstr "외부 포트" #: cps/templates/admin.html:129 cps/templates/config_view_edit.html:28 msgid "Books per Page" -msgstr "페이지당 보여 줄 책 수" +msgstr "페이지당 표시 개수" #: cps/templates/admin.html:133 msgid "Uploads" @@ -1706,27 +1676,27 @@ msgstr "공개 등록" #: cps/templates/admin.html:145 msgid "Magic Link Remote Login" -msgstr "Magic Link Remote Login" +msgstr "매직 링크 원격 로그인" #: cps/templates/admin.html:149 msgid "Reverse Proxy Login" -msgstr "Reverse Proxy Login" +msgstr "리버스 프록시 로그인" #: cps/templates/admin.html:154 cps/templates/config_edit.html:172 msgid "Reverse Proxy Header Name" -msgstr "Reverse Proxy Header Name" +msgstr "리버스 프록시 헤더 이름" #: cps/templates/admin.html:159 msgid "Edit Calibre Database Configuration" -msgstr "Calibre Database 환경 설정 편집" +msgstr "Calibre 데이터베이스 설정" #: cps/templates/admin.html:160 msgid "Edit Basic Configuration" -msgstr "기본 환경 설정 편집" +msgstr "기본 설정" #: cps/templates/admin.html:161 msgid "Edit UI Configuration" -msgstr "UI 환경 설정 편집" +msgstr "UI 설정" #: cps/templates/admin.html:167 msgid "Scheduled Tasks" @@ -1743,7 +1713,7 @@ msgstr "최대 기간" #: cps/templates/admin.html:178 cps/templates/schedule_edit.html:29 msgid "Generate Thumbnails" -msgstr "섬네일 생성성" +msgstr "섬네일 생성" #: cps/templates/admin.html:182 msgid "Generate series cover thumbnails" @@ -1752,15 +1722,15 @@ msgstr "시리즈 표지 섬네일 생성" #: cps/templates/admin.html:186 cps/templates/admin.html:208 #: cps/templates/schedule_edit.html:37 msgid "Reconnect Calibre Database" -msgstr "Calibre DB 다시 연결" +msgstr "Calibre 데이터베이스 다시 연결" #: cps/templates/admin.html:190 cps/templates/schedule_edit.html:41 msgid "Generate Metadata Backup Files" -msgstr "메타 정보 백업 파일 생성" +msgstr "메타데이터 백업" #: cps/templates/admin.html:197 msgid "Refresh Thumbnail Cache" -msgstr "섬네일 캐시 새로 고침침" +msgstr "섬네일 캐시 새로 고침" #: cps/templates/admin.html:203 msgid "Administration" @@ -1768,7 +1738,7 @@ msgstr "관리" #: cps/templates/admin.html:204 msgid "Download Debug Package" -msgstr "Debug Package 다운로드" +msgstr "디버그 패키지 다운로드" #: cps/templates/admin.html:205 msgid "View Logs" @@ -1808,7 +1778,7 @@ msgstr "업데이트 실행" #: cps/templates/admin.html:253 msgid "Are you sure you want to restart?" -msgstr "재시작을 하시겠습니까?" +msgstr "재시작하시겠습니까?" #: cps/templates/admin.html:258 cps/templates/admin.html:272 #: cps/templates/admin.html:292 cps/templates/config_db.html:82 @@ -1816,19 +1786,19 @@ msgid "OK" msgstr "예" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "취소" #: cps/templates/admin.html:271 msgid "Are you sure you want to shutdown?" -msgstr "종료를 하시겠습니까?" +msgstr "종료하시겠습니까?" #: cps/templates/admin.html:283 msgid "Updating, please do not reload this page" @@ -1836,51 +1806,111 @@ msgstr "업데이트 중입니다. 이 페이지를 새로고침 하지 마세 #: cps/templates/author.html:15 msgid "via" -msgstr "via" +msgstr "통해" #: cps/templates/author.html:23 msgid "In Library" -msgstr "In Library" +msgstr "서재에 있음" #: cps/templates/author.html:26 cps/templates/index.html:74 #: cps/templates/search.html:31 cps/templates/shelf.html:20 msgid "Sort according to book date, newest first" -msgstr "책 목록 정렬, 최신순" +msgstr "등록 최신순" #: cps/templates/author.html:27 cps/templates/index.html:75 #: cps/templates/search.html:32 cps/templates/shelf.html:21 msgid "Sort according to book date, oldest first" -msgstr "책 목록 정렬, 오래된 순" +msgstr "등록 오래된 순" #: cps/templates/author.html:28 cps/templates/index.html:76 #: cps/templates/search.html:33 cps/templates/shelf.html:22 msgid "Sort title in alphabetical order" -msgstr "가나다 순 정렬" +msgstr "이름순" #: cps/templates/author.html:29 cps/templates/index.html:77 #: cps/templates/search.html:34 cps/templates/shelf.html:23 msgid "Sort title in reverse alphabetical order" -msgstr "가나다 역순 정렬" +msgstr "이름 역순" #: cps/templates/author.html:30 cps/templates/index.html:80 #: cps/templates/search.html:37 cps/templates/shelf.html:26 msgid "Sort according to publishing date, newest first" -msgstr "발행일 순으로 정렬, 최신순" +msgstr "발행일 최신순" #: cps/templates/author.html:31 cps/templates/index.html:81 #: cps/templates/search.html:38 cps/templates/shelf.html:27 msgid "Sort according to publishing date, oldest first" -msgstr "발행일 순으로 정렬, 오래된 순" +msgstr "발행일 오래된 순" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" -msgstr "reduce" +msgstr "줄이기" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" -msgstr "More by" +msgstr "더보기" + +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "책 %(index)s / %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "언어" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "출판사" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "발행일" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "설명:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "이전" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "다음" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "검색 결과가 없습니다." + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "홈" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "검색" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "로그아웃" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" #: cps/templates/book_edit.html:11 msgid "Delete Book" @@ -1888,15 +1918,15 @@ msgstr "책 삭제" #: cps/templates/book_edit.html:14 msgid "Delete formats:" -msgstr "유형 삭제:" +msgstr "형식 삭제:" #: cps/templates/book_edit.html:25 msgid "Convert book format:" -msgstr "책 유형 변경:" +msgstr "책 형식 변경:" #: cps/templates/book_edit.html:30 msgid "Convert from:" -msgstr "변경전 유형:" +msgstr "변환 전 유형:" #: cps/templates/book_edit.html:32 cps/templates/book_edit.html:39 msgid "select an option" @@ -1904,105 +1934,113 @@ msgstr "옵션 선택" #: cps/templates/book_edit.html:37 msgid "Convert to:" -msgstr "변경후 유형:" +msgstr "변환 후 유형:" #: cps/templates/book_edit.html:46 msgid "Convert book" -msgstr "책 변경" +msgstr "책 변환" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "업로드 중" + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "닫기" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "오류" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "업로드 완료, 처리 중입니다. 잠시만 기다려 주세요." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "다른 형식 업로드" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "책 제목" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "저자" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 -msgid "Description" -msgstr "책 소개" +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "태그" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "시리즈 ID" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "발행일" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "평점" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 +msgid "Description" +msgstr "설명" + +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "식별자" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "식별자 유형" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "식별자 값" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "삭제" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "식별자 추가" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "태그" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "시리즈 ID" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "평점" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" -msgstr "URL로 표지 추가(JPEG - 이미지를 다운로드 하여 DB에 저장합니다)" +msgstr "URL로 표지 추가 (JPEG 이미지를 다운로드합니다)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" -msgstr "로컬 디스크에서 표지 업로드" +msgstr "내 컴퓨터에서 표지 업로드" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "출간일" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "출판사" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "언어" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "예" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "아니오" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "업로드 유형" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" -msgstr "저장 후 책 보기" +msgstr "저장 후 상세정보로" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" -msgstr "메타정보 가져오기" +msgstr "메타데이터 가져오기" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2010,45 +2048,39 @@ msgstr "메타정보 가져오기" msgid "Save" msgstr "저장" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "키워드" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "키워드 검색" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" -msgstr "표지를 클릭하여 양식에 메타데이터 로드" +msgstr "검색 결과를 사용하려면 표지를 클릭하세요." -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." -msgstr "불러오는 중..." +msgstr "불러오는 중" -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "닫기" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "출처" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" -msgstr "오류 검색!" +msgstr "오류가 발생했습니다!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." -msgstr "검색 결과가 없습니다! 다른 키워드를 시도하십시오." +msgstr "검색 결과가 없습니다. 다른 검색어로 검색해 보세요." #: cps/templates/book_table.html:12 cps/templates/book_table.html:69 #: cps/templates/user_table.html:14 cps/templates/user_table.html:77 #: cps/templates/user_table.html:100 msgid "This Field is Required" -msgstr "이 필드는 필수 항목입니다" +msgstr "이 항목은 필수입니다." #: cps/templates/book_table.html:37 msgid "Merge selected books" @@ -2056,11 +2088,11 @@ msgstr "선택한 책 병합" #: cps/templates/book_table.html:38 cps/templates/user_table.html:124 msgid "Remove Selections" -msgstr "선택 제거" +msgstr "선택 항목 삭제" #: cps/templates/book_table.html:41 msgid "Exchange author and title" -msgstr "저자 및 제목 교환" +msgstr "저자와 제목 바꾸기" #: cps/templates/book_table.html:47 msgid "Update Title Sort automatically" @@ -2068,7 +2100,7 @@ msgstr "자동으로 제목 정렬 업데이트" #: cps/templates/book_table.html:51 msgid "Update Author Sort automatically" -msgstr "자동으로 작성자 정렬 업데이트" +msgstr "자동으로 저자 정렬 업데이트" #: cps/templates/book_table.html:63 cps/templates/book_table.html:69 msgid "Enter Title" @@ -2109,7 +2141,7 @@ msgstr "시리즈 입력" #: cps/templates/book_table.html:69 msgid "Series Index" -msgstr "시리즈 색인" +msgstr "시리즈 순서" #: cps/templates/book_table.html:70 msgid "Enter Languages" @@ -2117,7 +2149,7 @@ msgstr "언어 입력" #: cps/templates/book_table.html:71 msgid "Publishing Date" -msgstr "출간일" +msgstr "발행일" #: cps/templates/book_table.html:72 msgid "Enter Publishers" @@ -2125,15 +2157,15 @@ msgstr "출판사 입력" #: cps/templates/book_table.html:73 msgid "Enter comments" -msgstr "코멘트 입력" +msgstr "설명 입력" #: cps/templates/book_table.html:73 msgid "Comments" -msgstr "코멘트" +msgstr "설명" #: cps/templates/book_table.html:75 msgid "Archive Status" -msgstr "아카이브 상태태" +msgstr "보관 상태" #: cps/templates/book_table.html:77 cps/templates/search_form.html:42 msgid "Read Status" @@ -2147,17 +2179,17 @@ msgid "Enter " msgstr "입력 " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" -msgstr "정말 확신합니까?" +msgstr "정말로 하시겠습니까?" #: cps/templates/book_table.html:117 msgid "Books with Title will be merged from:" -msgstr "제목이 있는 책은 다음에서 병합됨:" +msgstr "다음 책들을 병합합니다:" #: cps/templates/book_table.html:121 msgid "Into Book with Title:" -msgstr "제목이 있는 책으로:" +msgstr "다음 책으로 병합:" #: cps/templates/book_table.html:126 msgid "Merge" @@ -2165,27 +2197,27 @@ msgstr "병합" #: cps/templates/config_db.html:12 msgid "Location of Calibre Database" -msgstr "Calibre Database 위치" +msgstr "Calibre 데이터베이스 경로" #: cps/templates/config_db.html:21 msgid "Separate Book Files from Library" -msgstr "" +msgstr "책 저장 경로 분리" #: cps/templates/config_db.html:34 msgid "Use Google Drive?" -msgstr "Google 드라이브를 사용하시겠습니까?" +msgstr "Google 드라이브 사용" #: cps/templates/config_db.html:39 msgid "Authenticate Google Drive" -msgstr "구글 드라이브 인증" +msgstr "Google 드라이브 인증" #: cps/templates/config_db.html:44 msgid "Google Drive Calibre folder" -msgstr "구글 드라이브 Calibre 폴더" +msgstr "Google 드라이브 Calibre 폴더" #: cps/templates/config_db.html:52 msgid "Metadata Watch Channel ID" -msgstr "Metadata Watch Channel ID" +msgstr "Google 드라이브 API ID" #: cps/templates/config_db.html:55 msgid "Revoke" @@ -2193,55 +2225,55 @@ msgstr "취소" #: cps/templates/config_db.html:80 msgid "New db location is invalid, please enter valid path" -msgstr "잘못된 새 DB 위치. 올바른 경로 입력 필요" +msgstr "데이터베이스 경로가 올바르지 않습니다. 유효한 경로를 입력해주세요." #: cps/templates/config_edit.html:18 msgid "Server Configuration" -msgstr "서버 구성" +msgstr "서버 설정" #: cps/templates/config_edit.html:25 msgid "Server Port" -msgstr "Server Port" +msgstr "서버 포트" #: cps/templates/config_edit.html:28 msgid "SSL certfile location (leave it empty for non-SSL Servers)" -msgstr "SSL 인증서 파일 위치(비 SSL 서버의 경우 비워 둡니다)" +msgstr "SSL 인증서 파일 경로 (미사용 시 공란)" #: cps/templates/config_edit.html:35 msgid "SSL Keyfile location (leave it empty for non-SSL Servers)" -msgstr "SSL 키 파일 위치(비 SSL 서버의 경우 비워 둡니다)" +msgstr "SSL 키 파일 경로 (미사용시 공란)" #: cps/templates/config_edit.html:43 msgid "Update Channel" -msgstr "업데이트 취소" +msgstr "업데이트 채널" #: cps/templates/config_edit.html:45 msgid "Stable" -msgstr "안정버전" +msgstr "안정 버전" #: cps/templates/config_edit.html:46 msgid "Nightly" -msgstr "실험버전" +msgstr "실험 버전" #: cps/templates/config_edit.html:50 msgid "Trusted Hosts (Comma Separated)" -msgstr "신뢰할 수 있는 호스트(쉼표로 구분)" +msgstr "신뢰할 수 있는 호스트 (쉼표로 구분)" #: cps/templates/config_edit.html:61 msgid "Logfile Configuration" -msgstr "Logfile 설정" +msgstr "로그 파일 설정" #: cps/templates/config_edit.html:77 msgid "Location and name of logfile (calibre-web.log for no entry)" -msgstr "로그 파일의 위치 및 이름(항목이 없는 경우 calibre-web.log)" +msgstr "로그 파일 경로 및 파일명 (공란일 경우 calibre-web.log로 저장됩니다)" #: cps/templates/config_edit.html:82 msgid "Enable Access Log" -msgstr "액세스 로그 활성화" +msgstr "액세스 로그 사용" #: cps/templates/config_edit.html:85 msgid "Location and name of access logfile (access.log for no entry)" -msgstr "액세스 로그 파일의 위치 및 이름(항목이 없는 경우 access.log)" +msgstr "액세스 로그 파일 경로 및 파일명 (공란일 경우 access.log로 저장됩니다)" #: cps/templates/config_edit.html:96 msgid "Feature Configuration" @@ -2249,31 +2281,31 @@ msgstr "기능 설정" #: cps/templates/config_edit.html:104 msgid "Convert non-English characters in title and author while saving to disk" -msgstr "제목과 저자가 영어가 아닌 경우 원어로 저장하기" +msgstr "제목과 저자가 영문이 아닐 경우 영문(발음 그대로)으로 변환하여 저장" #: cps/templates/config_edit.html:108 msgid "Embed Metadata to Ebook File on Download/Conversion/e-mail (needs Calibre/Kepubify binaries)" -msgstr "" +msgstr "다운로드/변환/이메일 전송 시 전자책 파일에 메타데이터 포함 (Calibre/Kepubify 필요)" #: cps/templates/config_edit.html:112 msgid "Enable Uploads" -msgstr "업로드 활성화" +msgstr "업로드 사용" #: cps/templates/config_edit.html:112 msgid "(Please ensure that users also have upload permissions)" -msgstr "(사용자에게 업로드 권한도 있는지 확인하십시오)" +msgstr "(사용자에게 업로드 권한이 있어야 가능)" #: cps/templates/config_edit.html:116 msgid "Allowed Upload Fileformats" -msgstr "허용된 업로드 파일 형식" +msgstr "허용할 업로드 파일 형식" #: cps/templates/config_edit.html:122 msgid "Enable Anonymous Browsing" -msgstr "익명 탐색 활성화" +msgstr "익명 탐색 사용" #: cps/templates/config_edit.html:126 msgid "Enable Public Registration" -msgstr "공개 등록 활성화" +msgstr "공개 등록 사용" #: cps/templates/config_edit.html:131 msgid "Use Email as Username" @@ -2281,19 +2313,19 @@ msgstr "이메일을 사용자 이름으로 사용" #: cps/templates/config_edit.html:136 msgid "Enable Magic Link Remote Login" -msgstr "Enable Magic Link Remote Login" +msgstr "매직 링크 원격 로그인 사용" #: cps/templates/config_edit.html:141 msgid "Enable Kobo sync" -msgstr "Kobo sync 활성화" +msgstr "Kobo sync 사용" #: cps/templates/config_edit.html:146 msgid "Proxy unknown requests to Kobo Store" -msgstr "Kobo Store에 알 수 없는 요청 프록시" +msgstr "Kobo 스토어에 알 수 없는 요청 프록시 전달" #: cps/templates/config_edit.html:149 msgid "Server External Port (for port forwarded API calls)" -msgstr "서버 외부 포트(포트 전달 API 호출용)" +msgstr "서버 외부 포트 (포트 포워딩 API 호출용)" #: cps/templates/config_edit.html:157 msgid "Use Goodreads" @@ -2301,19 +2333,19 @@ msgstr "Goodreads 사용" #: cps/templates/config_edit.html:161 msgid "Goodreads API Key" -msgstr "Goodreads API Key" +msgstr "Goodreads API 키" #: cps/templates/config_edit.html:168 msgid "Allow Reverse Proxy Authentication" -msgstr "역방향 프록시 인증 허용" +msgstr "리버스 프록시 인증 허용" #: cps/templates/config_edit.html:179 msgid "Login type" -msgstr "로그인 유형" +msgstr "로그인 방식" #: cps/templates/config_edit.html:181 msgid "Use Standard Authentication" -msgstr "표준 인증 사용" +msgstr "기본 인증 사용" #: cps/templates/config_edit.html:183 msgid "Use LDAP Authentication" @@ -2345,15 +2377,15 @@ msgstr "SSL" #: cps/templates/config_edit.html:209 msgid "LDAP CACertificate Path (Only needed for Client Certificate Authentication)" -msgstr "LDAP CACertificate 경로(클라이언트 인증서 인증에만 필요)" +msgstr "LDAP CA 인증서 경로 (클라이언트 인증서 인증 시에만 필요)" #: cps/templates/config_edit.html:216 msgid "LDAP Certificate Path (Only needed for Client Certificate Authentication)" -msgstr "LDAP 인증서 경로(클라이언트 인증서 인증에만 필요)" +msgstr "LDAP 인증서 경로 (클라이언트 인증서 인증 시에만 필요)" #: cps/templates/config_edit.html:223 msgid "LDAP Keyfile Path (Only needed for Client Certificate Authentication)" -msgstr "LDAP 키 파일 경로(클라이언트 인증서 인증에만 필요)" +msgstr "LDAP 키 파일 경로 (클라이언트 인증서 인증 시에만 필요)" #: cps/templates/config_edit.html:232 msgid "LDAP Authentication" @@ -2365,15 +2397,15 @@ msgstr "익명" #: cps/templates/config_edit.html:235 msgid "Unauthenticated" -msgstr "인증되지 않음" +msgstr "미인증" #: cps/templates/config_edit.html:236 msgid "Simple" -msgstr "Simple" +msgstr "단순 인증" #: cps/templates/config_edit.html:241 msgid "LDAP Administrator Username" -msgstr "LDAP 관리자 사용자 이름" +msgstr "LDAP 관리자 아이디" #: cps/templates/config_edit.html:247 msgid "LDAP Administrator Password" @@ -2381,27 +2413,27 @@ msgstr "LDAP 관리자 비밀번호" #: cps/templates/config_edit.html:252 msgid "LDAP Distinguished Name (DN)" -msgstr "LDAP 고유 이름(DN)" +msgstr "LDAP 식별 이름 (DN)" #: cps/templates/config_edit.html:256 msgid "LDAP User Object Filter" -msgstr "LDAP 사용자 개체 필터" +msgstr "LDAP 사용자 객체 필터" #: cps/templates/config_edit.html:261 msgid "LDAP Server is OpenLDAP?" -msgstr "LDAP 서버는 OpenLDAP입니까?" +msgstr "OpenLDAP 서버인 경우 체크" #: cps/templates/config_edit.html:263 msgid "Following Settings are Needed For User Import" -msgstr "사용자 가져오기에는 다음 설정이 필요" +msgstr "사용자를 가져오기 위해 다음 설정이 필요합니다." #: cps/templates/config_edit.html:265 msgid "LDAP Group Object Filter" -msgstr "LDAP 그룹 개체 필터" +msgstr "LDAP 그룹 객체 필터" #: cps/templates/config_edit.html:269 msgid "LDAP Group Name" -msgstr "DAP 그룹 이름" +msgstr "LDAP 그룹 이름" #: cps/templates/config_edit.html:273 msgid "LDAP Group Members Field" @@ -2409,7 +2441,7 @@ msgstr "LDAP 그룹 구성원 필드" #: cps/templates/config_edit.html:277 msgid "LDAP Member User Filter Detection" -msgstr "LDAP 구성원 사용자 필터 감지" +msgstr "LDAP 사용자 감지 필터" #: cps/templates/config_edit.html:279 msgid "Autodetect" @@ -2417,7 +2449,7 @@ msgstr "자동 감지" #: cps/templates/config_edit.html:280 msgid "Custom Filter" -msgstr "맞춤 필터" +msgstr "사용자 지정 필터" #: cps/templates/config_edit.html:285 msgid "LDAP Member User Filter" @@ -2426,7 +2458,7 @@ msgstr "LDAP 구성원 사용자 필터" #: cps/templates/config_edit.html:296 #, python-format msgid "Obtain %(provider)s OAuth Credential" -msgstr "%(provider)s OAuth 자격 증명 얻기" +msgstr "%(provider)s OAuth 인증 정보 받기" #: cps/templates/config_edit.html:299 #, python-format @@ -2436,50 +2468,47 @@ msgstr "%(provider)s OAuth 클라이언트 ID" #: cps/templates/config_edit.html:303 #, python-format msgid "%(provider)s OAuth Client Secret" -msgstr "%(provider)s OAuth 클라이언트 암호" +msgstr "%(provider)s OAuth 클라이언트 비밀키" #: cps/templates/config_edit.html:319 msgid "External binaries" -msgstr "외부 바이너리" +msgstr "외부 바이너리 설정" #: cps/templates/config_edit.html:325 -#, fuzzy msgid "Path to Calibre Binaries" -msgstr "Calibre E-Book 변환기 경로" +msgstr "Calibre 바이너리 경로" #: cps/templates/config_edit.html:333 msgid "Calibre E-Book Converter Settings" -msgstr "Calibre E-Book 변환기 경로 설정" +msgstr "Calibre E-Book 컨버터 설정" #: cps/templates/config_edit.html:336 msgid "Path to Kepubify E-Book Converter" -msgstr "Kepubify 전자책 변환기 경로" +msgstr "Kepubify 전자책 컨버터 경로" #: cps/templates/config_edit.html:344 -#, fuzzy msgid "Location of Unrar binary" -msgstr "UnRar 바이너리의 위치" +msgstr "UnRar 바이너리 경로" #: cps/templates/config_edit.html:360 -#, fuzzy msgid "Security Settings" -msgstr "OAuth 설정" +msgstr "보안 설정" #: cps/templates/config_edit.html:368 msgid "Limit failed login attempts" -msgstr "로그인 시도 제한" +msgstr "로그인 실패 횟수 제한" #: cps/templates/config_edit.html:372 msgid "Configure Backend for Limiter" -msgstr "" +msgstr "Flask-Limiter 데이터베이스 주소" #: cps/templates/config_edit.html:376 msgid "Options for Limiter Backend" -msgstr "" +msgstr "Flask-Limiter 옵션" #: cps/templates/config_edit.html:382 msgid "Check if file extensions matches file content on upload" -msgstr "" +msgstr "업로드 시 파일 형식 일치 검사" #: cps/templates/config_edit.html:385 msgid "Session protection" @@ -2491,12 +2520,11 @@ msgstr "기본" #: cps/templates/config_edit.html:388 msgid "Strong" -msgstr "강함함" +msgstr "강력" #: cps/templates/config_edit.html:393 -#, fuzzy msgid "User Password policy" -msgstr "사용자 비밀번호 정책" +msgstr "비밀번호 강도 규칙 사용" #: cps/templates/config_edit.html:397 msgid "Minimum password length" @@ -2504,27 +2532,27 @@ msgstr "최소 비밀번호 길이" #: cps/templates/config_edit.html:402 msgid "Enforce number" -msgstr "반드시 숫자 입력" +msgstr "숫자 필수" #: cps/templates/config_edit.html:406 msgid "Enforce lowercase characters" -msgstr "반드시 소문자 입력" +msgstr "소문자 필수" #: cps/templates/config_edit.html:410 msgid "Enforce uppercase characters" -msgstr "반드시 대문자 입력" +msgstr "대문자 필수" #: cps/templates/config_edit.html:414 msgid "Enforce characters (needed For Chinese/Japanese/Korean Characters)" -msgstr "" +msgstr "한자/일본어/한국어 문자 필수" #: cps/templates/config_edit.html:418 msgid "Enforce special characters" -msgstr "반드시 특수문자 입력력" +msgstr "특수문자 필수" #: cps/templates/config_view_edit.html:17 msgid "View Configuration" -msgstr "환경 설정 보기" +msgstr "환경 설정" #: cps/templates/config_view_edit.html:32 msgid "No. of Random Books to Display" @@ -2532,7 +2560,7 @@ msgstr "랜덤 보기시 표시할 책 수" #: cps/templates/config_view_edit.html:36 msgid "No. of Authors to Display Before Hiding (0=Disable Hiding)" -msgstr "숨기기 전에 표시할 저자 수(0=숨기기 비활성화)" +msgstr "숨기기 전 표시할 저자 수 (0=숨기기 비활성화)" #: cps/templates/config_view_edit.html:40 cps/templates/readcbr.html:101 msgid "Theme" @@ -2544,7 +2572,7 @@ msgstr "기본 테마" #: cps/templates/config_view_edit.html:43 msgid "caliBlur! Dark Theme" -msgstr "caliBlur! 다크 모드" +msgstr "caliBlur! 다크 테마" #: cps/templates/config_view_edit.html:47 msgid "Regular Expression for Ignoring Columns" @@ -2552,19 +2580,19 @@ msgstr "열을 무시하는 정규식" #: cps/templates/config_view_edit.html:51 msgid "Link Read/Unread Status to Calibre Column" -msgstr "읽음/읽지 않음 상태를 Calibre 열에 연결" +msgstr "읽음/읽지 않음 상태를 Calibre 데이터에 저장" #: cps/templates/config_view_edit.html:60 msgid "View Restrictions based on Calibre column" -msgstr "Calibre 열을 기반으로 한 보기 제한" +msgstr "Calibre 데이터 기반 화면 표시 제한" #: cps/templates/config_view_edit.html:69 msgid "Regular Expression for Title Sorting" -msgstr "제목 정렬을 위한 정규식" +msgstr "제목 정렬 정규식" #: cps/templates/config_view_edit.html:80 msgid "Default Settings for New Users" -msgstr "새 사용자를 위한 기본 설정" +msgstr "새 사용자 기본 설정" #: cps/templates/config_view_edit.html:88 cps/templates/user_edit.html:96 msgid "Admin User" @@ -2608,12 +2636,12 @@ msgstr "책의 기본 표시 언어" #: cps/templates/config_view_edit.html:147 msgid "Default Visibilities for New Users" -msgstr "신규 사용자를 위한 기본 기능" +msgstr "새 사용자 기본 기능" #: cps/templates/config_view_edit.html:163 cps/templates/user_edit.html:84 #: cps/templates/user_table.html:154 msgid "Show Random Books in Detail View" -msgstr "상세 보기에서 임의의 책 표시" +msgstr "목록 보기에서 무작위 추천 표시" #: cps/templates/config_view_edit.html:166 cps/templates/user_edit.html:87 msgid "Add Allowed/Denied Tags" @@ -2623,98 +2651,81 @@ msgstr "허용/거부 태그 추가" msgid "Add Allowed/Denied custom column values" msgstr "허용/거부 사용자 정의 열 값 추가" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" -msgstr "부라우저에서 보기" +msgstr "웹에서 보기" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" -msgstr "브라우저에서 듣기" +msgstr "웹에서 듣기" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "책 %(index)s의 %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "출간일" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" -msgstr "읽지 않은 상태로 표시" +msgstr "읽지 않음으로 표시" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" -msgstr "읽은 상태로 표시" +msgstr "읽음으로 표시" -#: cps/templates/detail.html:254 -#, fuzzy +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" -msgstr "읽지 않은 상태로 표시" +msgstr "읽음/읽지 않음으로 표시" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" -msgstr "읽기" +msgstr "읽음" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" -msgstr "아카이브에서 복원" +msgstr "보관함에서 복원" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" -msgstr "아카이브에 추가" +msgstr "보관함에 추가" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" -msgstr "" +msgstr "책을 보관하면 Calibre-Web에서 숨겨지고 Kobo 리더에서 삭제됩니다." -#: cps/templates/detail.html:267 -#, fuzzy +#: cps/templates/detail.html:275 msgid "Archive" -msgstr "보관됨" +msgstr "보관" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "설명:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "서재에 추가" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(공개)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" -msgstr "메타정보보 편집" +msgstr "메타데이터 편집" #: cps/templates/email_edit.html:13 msgid "Email Account Type" -msgstr "서버 유형 선택" +msgstr "이메일 유형 선택" #: cps/templates/email_edit.html:15 -#, fuzzy msgid "Standard Email Account" -msgstr "표준 이메일 계정 사용" +msgstr "표준 이메일 계정" #: cps/templates/email_edit.html:16 -#, fuzzy msgid "Gmail Account" msgstr "Gmail 계정" #: cps/templates/email_edit.html:22 msgid "Setup Gmail Account" -msgstr "Gmail 계정 설정정" +msgstr "Gmail 계정 설정" #: cps/templates/email_edit.html:24 msgid "Revoke Gmail Access" -msgstr "Gmail 액세스 취소" +msgstr "Gmail 접근 권한 해제" #: cps/templates/email_edit.html:42 msgid "STARTTLS" @@ -2730,12 +2741,11 @@ msgstr "SMTP 비밀번호" #: cps/templates/email_edit.html:58 msgid "Attachment Size Limit" -msgstr "첨부 파일 크기 제한" +msgstr "첨부파일 최대 크기" #: cps/templates/email_edit.html:66 -#, fuzzy msgid "Save and Send Test Email" -msgstr "저장 후 테스트 이메일 보내기" +msgstr "저장 및 테스트 메일 전송" #: cps/templates/email_edit.html:70 cps/templates/layout.html:26 #: cps/templates/shelf_order.html:42 cps/templates/user_table.html:174 @@ -2744,7 +2754,7 @@ msgstr "뒤로" #: cps/templates/email_edit.html:74 msgid "Allowed Domains (Whitelist)" -msgstr "허용하는 도메인(화이트 리스트)" +msgstr "허용하는 도메인 (화이트 리스트)" #: cps/templates/email_edit.html:78 cps/templates/email_edit.html:105 msgid "Add Domain" @@ -2761,20 +2771,15 @@ msgstr "도메인 입력" #: cps/templates/email_edit.html:92 msgid "Denied Domains (Blacklist)" -msgstr "거부 도메인(블랙리스트)" - -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "다음" +msgstr "차단 도메인 (블랙리스트)" #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" -msgstr "텍스트 편집기에서 .kobo/Kobo/Kobo eReader.conf 파일을 열고 다음을 추가(또는 편집):" +msgstr "텍스트 편집기에서 .kobo/Kobo/Kobo eReader.conf 파일을 열고 다음을 추가하거나 수정하세요:" #: cps/templates/generate_kobo_auth_url.html:11 -#, fuzzy msgid "Kobo Token:" -msgstr "코보 연동 토큰" +msgstr "Kobo 토큰:" #: cps/templates/grid.html:21 msgid "List" @@ -2782,77 +2787,82 @@ msgstr "목록" #: cps/templates/http_error.html:34 msgid "Calibre-Web Instance is unconfigured, please contact your administrator" -msgstr "Calibre-Web Instance가 구성되지 않았습니다. 관리자에게 문의" +msgstr "Calibre-Web이 설정되지 않았습니다. 관리자에게 문의하세요." #: cps/templates/http_error.html:44 msgid "Create Issue" msgstr "이슈 생성" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "데이터베이스 설정" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "홈으로" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "로그아웃" #: cps/templates/index.html:71 msgid "Sort ascending according to download count" -msgstr "다운로드 횟수에 따라 오름차순 정렬" +msgstr "다운로드 수 오름차순 정렬" #: cps/templates/index.html:72 msgid "Sort descending according to download count" -msgstr "다운로드 횟수에 따라 내림차순 정렬" +msgstr "다운로드 수 내림차순 정렬" #: cps/templates/index.html:78 cps/templates/search.html:35 #: cps/templates/shelf.html:24 msgid "Sort authors in alphabetical order" -msgstr "저자를 가나다순(알파벳순)으로 정렬" +msgstr "저자명 오름차순 정렬" #: cps/templates/index.html:79 cps/templates/search.html:36 #: cps/templates/shelf.html:25 msgid "Sort authors in reverse alphabetical order" -msgstr "저자를 가나다(알파벳) 역순으로 정렬" +msgstr "저자명 내림차순 정렬" #: cps/templates/index.html:83 msgid "Sort ascending according to series index" -msgstr "시리즈 인덱스에 따라 오름차순 정렬" +msgstr "시리즈 순서 오름차순 정렬" #: cps/templates/index.html:84 msgid "Sort descending according to series index" -msgstr "시리즈 인덱스에 따라 내림차순 정렬" +msgstr "시리즈 순서 내림차순 정렬" #: cps/templates/index.xml:7 msgid "Start" -msgstr "시작" +msgstr "시작하기" #: cps/templates/index.xml:19 msgid "Alphabetical Books" -msgstr "가나다(알파벳) 순" +msgstr "책 이름 오름차순 정렬" #: cps/templates/index.xml:23 msgid "Books sorted alphabetically" -msgstr "가나다(알파벳)순으로 정렬한 책" +msgstr "책 이름 오름차순 정렬" #: cps/templates/index.xml:31 msgid "Popular publications from this catalog based on Downloads." -msgstr "다운로드 기준 인기 도서." +msgstr "다운로드 기준 인기 책" #: cps/templates/index.xml:40 msgid "Popular publications from this catalog based on Rating." -msgstr "별점 기준 인기 도서." +msgstr "평점 기준 인기 책" #: cps/templates/index.xml:45 msgid "Recently added Books" -msgstr "최근 추가된 도서" +msgstr "최근 추가된 책" #: cps/templates/index.xml:49 msgid "The latest Books" -msgstr "최신 도서" +msgstr "최신 책" #: cps/templates/index.xml:54 msgid "Random Books" -msgstr "랜덤 정렬" +msgstr "랜덤 책" #: cps/templates/index.xml:83 msgid "Books ordered by Author" @@ -2880,9 +2890,9 @@ msgstr "평점별 정렬" #: cps/templates/index.xml:137 msgid "Books ordered by file formats" -msgstr "파일 종류별 정렬" +msgstr "파일 형식별 정렬" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "서재" @@ -2891,66 +2901,43 @@ msgstr "서재" msgid "Books organized in shelves" msgstr "서재별 정렬" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "홈" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" -msgstr "탐색 활성화" +msgstr "메뉴 열기/닫기" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "검색" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "단순 인증" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "계정" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "로그아웃" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "업로드 중..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "오류" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "업로드 완료, 처리 중입니다. 잠시만 기다려 주십시오..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "설정" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" -msgstr "페이지를 새로고침 하지 마세요" +msgstr "페이지를 새로고침하지 마세요." -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "탐색" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "서재 정보" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "이전" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "책 상세정보" #: cps/templates/list.html:22 msgid "Grid" -msgstr "그리드드" +msgstr "그리드" #: cps/templates/listenmp3.html:167 msgid "Archived" @@ -2966,7 +2953,7 @@ msgstr "비밀번호를 잊으셨나요?" #: cps/templates/login.html:34 msgid "Log in with Magic Link" -msgstr "Magic Link로 로그인" +msgstr "매직 링크로 로그인" #: cps/templates/logviewer.html:6 msgid "Show Calibre-Web Log: " @@ -2990,7 +2977,7 @@ msgstr "Calibre-Web 로그 다운로드" #: cps/templates/logviewer.html:21 msgid "Download Access Log" -msgstr "Access Log 다운로드" +msgstr "액세스 로그 다운로드" #: cps/templates/modal_dialogs.html:6 msgid "Select Allowed/Denied Tags" @@ -2998,7 +2985,7 @@ msgstr "허용/거부 태그 선택" #: cps/templates/modal_dialogs.html:7 msgid "Select Allowed/Denied Custom Column Values" -msgstr "허용/거부 사용자 정의 열 값 선택" +msgstr "허용/거부 사용자 정의 항목 선택" #: cps/templates/modal_dialogs.html:8 msgid "Select Allowed/Denied Tags of User" @@ -3006,7 +2993,7 @@ msgstr "사용자의 허용/거부 태그 선택" #: cps/templates/modal_dialogs.html:9 msgid "Select Allowed/Denied Custom Column Values of User" -msgstr "사용자의 사용자 정의 열 값 허용/거부 선택" +msgstr "사용자 정의 항목 허용/거부 선택" #: cps/templates/modal_dialogs.html:15 msgid "Enter Tag" @@ -3018,23 +3005,23 @@ msgstr "보기 제한 추가" #: cps/templates/modal_dialogs.html:50 msgid "This book format will be permanently erased from database" -msgstr "이 책 형식은 데이터베이스에서 영구적으로 삭제됨" +msgstr "이 책의 형식은 데이터베이스에서 삭제됩니다." #: cps/templates/modal_dialogs.html:51 msgid "This book will be permanently erased from database" -msgstr "이 책은 데이터베이스에서 영구적으로 삭제됨" +msgstr "이 책은 데이터베이스에서 삭제됩니다." #: cps/templates/modal_dialogs.html:52 msgid "and hard disk" -msgstr "그리고 하드디스크" +msgstr "그리고 디스크에서도 삭제됩니다." #: cps/templates/modal_dialogs.html:56 msgid "Important Kobo Note: deleted books will remain on any paired Kobo device." -msgstr "Kobo 참고 사항: 삭제된 책은 페어링된 모든 Kobo 기기에 남아 있습니다." +msgstr "Kobo 알림: 삭제된 책은 연결된 Kobo 기기에 남아있습니다." #: cps/templates/modal_dialogs.html:57 msgid "Books must first be archived and the device synced before a book can safely be deleted." -msgstr "책을 안전하게 삭제하려면 먼저 책을 보관하고 기기를 동기화해야 합니다." +msgstr "책을 안전하게 삭제하려면 먼저 보관 처리 후 기기 동기화가 필요합니다." #: cps/templates/modal_dialogs.html:76 msgid "Choose File Location" @@ -3060,46 +3047,86 @@ msgstr "상위 폴더" msgid "Select" msgstr "선택" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" -msgstr "Ok" +msgstr "확인" #: cps/templates/osd.xml:5 msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web 책 목록" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "epub 리더" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "사용자 이름 선택" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" -msgstr "밝게" +msgstr "밝은색" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" -msgstr "어둡게" +msgstr "어두운색" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "세피아" -#: cps/templates/read.html:84 -#, fuzzy +#: cps/templates/read.html:90 msgid "Black" -msgstr "뒤로" +msgstr "검은색" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." -msgstr "사이드바가 열려 있을 때 텍스트 다시 배열." +msgstr "사이드바가 열릴 경우 텍스트를 재정렬합니다." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" -msgstr "글자 크기기" +msgstr "글자 크기" + +#: cps/templates/read.html:105 +msgid "Font" +msgstr "글꼴" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "기본값" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "Yahei" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "SimSun" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "KaiTi" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "페이지 보기" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "두쪽 보기" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "한쪽 보기" #: cps/templates/readcbr.html:8 msgid "Comic Reader" -msgstr "코믹 리더" +msgstr "만화 리더" #: cps/templates/readcbr.html:75 msgid "Keyboard Shortcuts" @@ -3115,11 +3142,11 @@ msgstr "다음 페이지" #: cps/templates/readcbr.html:80 msgid "Single Page Display" -msgstr "한 페이지 보기기" +msgstr "한 페이지 보기" #: cps/templates/readcbr.html:81 msgid "Long Strip Display" -msgstr "Long Strip Display" +msgstr "연속 보기" #: cps/templates/readcbr.html:82 msgid "Scale to Best" @@ -3127,15 +3154,15 @@ msgstr "최적 크기" #: cps/templates/readcbr.html:83 msgid "Scale to Width" -msgstr "폭 조절" +msgstr "너비에 맞춤" #: cps/templates/readcbr.html:84 msgid "Scale to Height" -msgstr "높이 조절" +msgstr "높이에 맞춤" #: cps/templates/readcbr.html:85 msgid "Scale to Native" -msgstr "기본 크기" +msgstr "원본 크기" #: cps/templates/readcbr.html:86 msgid "Rotate Right" @@ -3147,24 +3174,23 @@ msgstr "왼쪽으로 회전" #: cps/templates/readcbr.html:88 msgid "Flip Image" -msgstr "이미지 뒤집기" +msgstr "이미지 반전" #: cps/templates/readcbr.html:110 msgid "Display" -msgstr "화면" +msgstr "표시 방식" #: cps/templates/readcbr.html:113 -#, fuzzy msgid "Single Page" -msgstr "관리자 페이지" +msgstr "단일 페이지" #: cps/templates/readcbr.html:114 msgid "Long Strip" -msgstr "Long Strip" +msgstr "세로 스크롤" #: cps/templates/readcbr.html:119 msgid "Scale" -msgstr "크기" +msgstr "크기 조절" #: cps/templates/readcbr.html:122 msgid "Best" @@ -3180,7 +3206,7 @@ msgstr "높이" #: cps/templates/readcbr.html:125 msgid "Native" -msgstr "기본" +msgstr "원본 크기" #: cps/templates/readcbr.html:130 msgid "Rotate" @@ -3212,11 +3238,11 @@ msgstr "오른쪽에서 왼쪽" #: cps/templates/readcbr.html:162 msgid "Reset to Top" -msgstr "" +msgstr "맨 위로 이동" #: cps/templates/readcbr.html:163 msgid "Remember Position" -msgstr "" +msgstr "위치 기억하기" #: cps/templates/readcbr.html:168 msgid "Scrollbar" @@ -3244,7 +3270,7 @@ msgstr "텍스트 리더" #: cps/templates/register.html:4 msgid "Register New Account" -msgstr "새 계정 등록" +msgstr "새 사용자 등록" #: cps/templates/register.html:10 msgid "Choose a username" @@ -3256,27 +3282,23 @@ msgstr "이메일 주소" #: cps/templates/remote_login.html:5 msgid "Magic Link - Authorise New Device" -msgstr "Magic Link - 새 장치 승인" +msgstr "매직 링크 - 새 장치 승인" #: cps/templates/remote_login.html:7 msgid "On another device, login and visit:" -msgstr "다른 기기에서는 로그인 하고 방문:" +msgstr "다른 기기에서 로그인하여 방문하세요:" #: cps/templates/remote_login.html:11 msgid "Once verified, you will automatically be logged in on this device." -msgstr "승인이 되면 이 장치에 자동으로 로그인됩니다." +msgstr "승인되면 자동으로 로그인 됩니다." #: cps/templates/remote_login.html:14 msgid "This verification link will expire in 10 minutes." -msgstr "이 확인 링크는 10분 후에 만료됩니다." +msgstr "인증 링크는 10분 후 만료됩니다." #: cps/templates/schedule_edit.html:33 msgid "Generate Series Cover Thumbnails" -msgstr "시리즈 섬네일 표지 생성" - -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "결과 없음" +msgstr "시리즈 미리보기 표지 생성" #: cps/templates/search.html:7 msgid "Search Term:" @@ -3288,19 +3310,19 @@ msgstr "결과:" #: cps/templates/search_form.html:21 msgid "Published Date From" -msgstr "출간일(부터)" +msgstr "발행일(시작)" #: cps/templates/search_form.html:31 msgid "Published Date To" -msgstr "출간일(까지)" +msgstr "발행일(종료)" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" -msgstr "" +msgstr "모두" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" -msgstr "" +msgstr "없음" #: cps/templates/search_form.html:60 msgid "Exclude Tags" @@ -3334,13 +3356,15 @@ msgstr "평점(이상)" msgid "Rating Below" msgstr "평점(이하)" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" -msgstr "부터:" +msgstr "시작:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" -msgstr "까지:" +msgstr "종료:" #: cps/templates/shelf.html:13 msgid "Delete this Shelf" @@ -3362,6 +3386,14 @@ msgstr "정렬 변경 비활성화" msgid "Enable Change order" msgstr "정렬 변경 활성화" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "추가된 기준으로 최신순" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "추가된 기준으로 오래된 순" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "모두와 공유" @@ -3376,11 +3408,11 @@ msgstr "드래그하여 순서 변경" #: cps/templates/shelf_order.html:33 msgid "Hidden Book" -msgstr "숨겨진 책" +msgstr "숨긴 책" #: cps/templates/stats.html:7 msgid "Library Statistics" -msgstr "Calibre 서재 통계" +msgstr "서재 통계" #: cps/templates/stats.html:12 msgid "Books in this Library" @@ -3404,7 +3436,7 @@ msgstr "시스템 통계" #: cps/templates/stats.html:33 msgid "Program" -msgstr "프로그램램" +msgstr "프로그램" #: cps/templates/stats.html:34 msgid "Installed Version" @@ -3424,28 +3456,37 @@ msgstr "상태" #: cps/templates/tasks.html:16 msgid "Progress" -msgstr "Progress" +msgstr "진행률" #: cps/templates/tasks.html:17 msgid "Run Time" msgstr "실행 시간" -#: cps/templates/tasks.html:20 -msgid "Actions" -msgstr "Actions" +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "메시지" -#: cps/templates/tasks.html:40 -msgid "This task will be cancelled. Any progress made by this task will be saved." -msgstr "이 작업을 취소합니다. 이 작업의 모든 진행사항은 반영됩니다." +#: cps/templates/tasks.html:21 +msgid "Actions" +msgstr "작업" #: cps/templates/tasks.html:41 +msgid "This task will be cancelled. Any progress made by this task will be saved." +msgstr "작업이 취소됩니다. 진행된 부분은 저장됩니다." + +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." -msgstr "이 작업이 예약된 작업이라면, 다음 예약 시간에 다시 실행됩니다." +msgstr "예약된 작업일 경우, 다음 예약 시간에 이어서 실행됩니다." #: cps/templates/user_edit.html:20 msgid "Reset user Password" msgstr "사용자 비밀번호 초기화" +#: cps/templates/user_edit.html:28 +#, fuzzy +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "전자책 리더로 전송할 이메일 (여러 전자책 리더를 사용할 경우 쉼표로 구분하여 입력)" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "책 언어" @@ -3456,15 +3497,15 @@ msgstr "OAuth 설정" #: cps/templates/user_edit.html:56 msgid "Link" -msgstr "링크" +msgstr "연결" #: cps/templates/user_edit.html:58 msgid "Unlink" -msgstr "링크 해제" +msgstr "연결 해제" #: cps/templates/user_edit.html:64 msgid "Kobo Sync Token" -msgstr "코보 연동 토큰" +msgstr "Kobo 동기화 토큰" #: cps/templates/user_edit.html:66 msgid "Create/View" @@ -3472,15 +3513,15 @@ msgstr "생성/보기" #: cps/templates/user_edit.html:70 msgid "Force full kobo sync" -msgstr "강제로 코보 전체 동기화" +msgstr "강제 Kobo 전체 동기화" #: cps/templates/user_edit.html:88 msgid "Add allowed/Denied Custom Column Values" -msgstr "허용/거부 사용자 정의 열 값 추가" +msgstr "허용/거부할 사용자 정의 항목 추가" #: cps/templates/user_edit.html:137 msgid "Sync only books in selected shelves with Kobo" -msgstr "선택한 서가에 있는 책만 Kobo와 동기화" +msgstr "선택한 서재의 책만 Kobo와 동기화" #: cps/templates/user_edit.html:147 cps/templates/user_table.html:169 msgid "Delete User" @@ -3488,11 +3529,11 @@ msgstr "사용자 삭제" #: cps/templates/user_edit.html:159 msgid "Generate Kobo Auth URL" -msgstr "Kobo Auth URL 생성" +msgstr "Kobo 인증 URL 생성" #: cps/templates/user_table.html:80 cps/templates/user_table.html:103 msgid "Select..." -msgstr "선택..." +msgstr "선택" #: cps/templates/user_table.html:131 msgid "Edit User" @@ -3500,22 +3541,19 @@ msgstr "사용자 편집" #: cps/templates/user_table.html:134 msgid "Enter Username" -msgstr "사용자 이름" +msgstr "사용자 이름 입력" #: cps/templates/user_table.html:135 -#, fuzzy msgid "Enter Email" -msgstr "테스트 이메일" +msgstr "이메일 입력" #: cps/templates/user_table.html:136 -#, fuzzy msgid "Enter eReader Email" -msgstr "킨들로 보내기 이메일 주소" +msgstr "전자책 리더 이메일 입력" #: cps/templates/user_table.html:136 -#, fuzzy msgid "eReader Email" -msgstr "테스트 이메일" +msgstr "전자책 리더 이메일" #: cps/templates/user_table.html:137 msgid "Locale" @@ -3527,35 +3565,35 @@ msgstr "표시할 책 언어" #: cps/templates/user_table.html:139 msgid "Edit Allowed Tags" -msgstr "허용된 태그 편집" +msgstr "허용 태그 편집" #: cps/templates/user_table.html:139 msgid "Allowed Tags" -msgstr "허용된 태그" +msgstr "허용 태그" #: cps/templates/user_table.html:140 msgid "Edit Denied Tags" -msgstr "거부된 태그 편집" +msgstr "제외 태그 편집" #: cps/templates/user_table.html:140 msgid "Denied Tags" -msgstr "거부된 태그" +msgstr "제외 태그" #: cps/templates/user_table.html:141 msgid "Edit Allowed Column Values" -msgstr "허용된 열 값 편집" +msgstr "허용 항목 편집" #: cps/templates/user_table.html:141 msgid "Allowed Column Values" -msgstr "허용되는 열 값" +msgstr "허용 항목" #: cps/templates/user_table.html:142 msgid "Edit Denied Column Values" -msgstr "거부된 열 값 편집" +msgstr "제외 항목 편집" #: cps/templates/user_table.html:142 msgid "Denied Column Values" -msgstr "거부된 열 값" +msgstr "제외 항목" #: cps/templates/user_table.html:144 msgid "Change Password" @@ -3571,10 +3609,9 @@ msgstr "공개 서재 편집" #: cps/templates/user_table.html:152 msgid "Sync selected Shelves with Kobo" -msgstr "선택한 서재를 코보와 동기화" +msgstr "선택한 서재 Kobo 동기화" #: cps/templates/user_table.html:156 -#, fuzzy msgid "Show Read/Unread Section" -msgstr "읽음/읽지 않음 선택 표시" +msgstr "읽음/읽지 않음 표시" diff --git a/cps/translations/nl/LC_MESSAGES/messages.mo b/cps/translations/nl/LC_MESSAGES/messages.mo index 3d17b476..e8bde621 100644 Binary files a/cps/translations/nl/LC_MESSAGES/messages.mo and b/cps/translations/nl/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/nl/LC_MESSAGES/messages.po b/cps/translations/nl/LC_MESSAGES/messages.po index 92b4306f..5f7aadad 100644 --- a/cps/translations/nl/LC_MESSAGES/messages.po +++ b/cps/translations/nl/LC_MESSAGES/messages.po @@ -8,7 +8,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web (GPLV3)\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2023-12-20 22:00+0100\n" "Last-Translator: Michiel Cornelissen \n" "Language: nl\n" @@ -17,511 +17,529 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistieken" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "De server is herstart, vernieuw de pagina" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Bezig met het afsluiten van de server, sluit het venster" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "Gelukt! Database opnieuw verbonden" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Onbekende opdracht" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Test E-Mail wordt verzonden naar %(email)s, controleer de taken voor het resultaat" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Onbekend" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Systeembeheer" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Basisconfiguratie" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Uiterlijk aanpassen" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Aangepaste kolom Nr.%(column)d bestaat niet in de Calibre Database" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Systeembeheerder" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Alles" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Gebruiker niet gevonden" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} gebruikers succesvol verwijderd" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Alle talen" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Misvormd verzoek" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Gast naam kan niet worden veranderd" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Gast kan deze rol niet hebben" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Kan systeembeheerder rol niet verwijderen van de laatste systeembeheerder" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Waarde moet Waar of Onwaar zijn" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Ongeldige rol" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Gast kan dit niet bekijken" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Ongeldige waarde" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Gasts locale is automatisch bepaald en kan niet handmatig worden ingesteld" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Geen geldige locale is opgegeven" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Geen geldige boek taal is opgegeven" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parameter is niet gevonden" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Ongeldige gelezen kolom" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Ongeldige beperkte kolom" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web-configuratie bijgewerkt" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Wil je je Kobo Token echt verwijderen?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Wil je dit domein echt verwijderen?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Wil je deze gebruiker echt verwijderen?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Weet je zeker dat je deze boekenplank wilt verwijderen?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Weet je zeker dat je de locales van de geselecteerde gebruiker(s) wil veranderen?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Weet je zeker dat je de zichtbare talen voor de geselecteerde gebruiker(s) wil veranderen?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Weet je zeker dat je de geselecteerde rol van de geselecteerde gebruiker(s) wil veranderen?" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Weet je zeker dat je de geselecteerde beperkingen voor de geselecteerde gebruikers(s) wil verwijderen?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Weet je zeker dat je de geselecteerde zichtbaarheidsbeperkingen voor de geselecteerde gebruiker(s) wil veranderen?" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Weet je zeker dat je de synchronisatiegedrag van boekenplanken voor de geselecteerde gebruiker(s) wil veranderen?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Weet je zeker dat je de locatie van de Calibre-bibliotheek wil veranderen?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calibre-web gaat zoeken naar bijgewerkte omslagen en miniaturen bijwerken, dit kan even duren?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Weet u zeker dat u de volledige Calibre-Web synchronisatiedatabase wilt verwijderen om een volledige synchronisatie met uw Kobo Reader te forceren?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Weigeren" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Toestaan" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} synchronisatie objecten verwijderd" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Tag niet gevonden" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Ongeldige actie" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json is niet geconfigureerd voor webapplicatie" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "De locatie van het logbestand is onjuist, voer een geldige locatie in" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "De locatie vam het toegangslog is onjuist, voer een geldige locatie in" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Voer alsjeblieft een LDAP Provider, Port, DN en User Object Identifier in" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Voer een geldig LDAP Service Account en wachtwoord in" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Voer een LDAP Service Account in" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP Groep Object Filter Moet Een \"%s\" Formaat Identificiatie hebben" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAP Groep Object Filter heeft een niet-gebalanceerd haakje" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP Gebruiker Object Filter moet \"%s\" Formaat Identificatie hebben" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAP Gebruiker Filter heeft een niet-gebalanceerd haakje" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP Lid Gebruiker Filter moet een \"%s\" Formaat Identificatie hebben" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "LDAP Lid Gebruiker Filter heeft een niet-gebalanceerd haakje" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CACertficaat, Certificaat of Sleutel Locatie is ongeldig. Voer alsjeblieft een geldig pad in." -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Gebruiker toevoegen" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "SMTP-instellingen bewerken" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "Gelukt! Gmail account geverifieerd." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Database fout: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Test E-Mail wordt verzonden naar %(email)s, controleer de taken voor het resultaat" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Fout opgetreden bij het versturen van de test-e-mail: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Gelieve eerst je e-mail adres configureren..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "E-mailserver-instellingen bijgewerkt" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Bewerk instellingen van geplande taken" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "De starttijd van de taak is ongeldig" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "De duur van de taak is ongeldig" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Instellingen van geplande taken bijgewerkt" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Onbekende fout opgetreden. Probeer het later nog eens." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Instellingen database is niet schrijfbaar." -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Gebruiker '%(nick)s' bewerken" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Wachtwoord voor gebruiker %(user)s is hersteld" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Stel eerst SMTP-mail in..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Logbestand lezer" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Update opvragen" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Update downloaden" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Update uitpakken" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Update toepassen" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Databaseverbindingen zijn gesloten" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Bezig met stoppen van Calibre-Web" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Update voltooid, klik op 'Oké' en vernieuw de pagina" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Update mislukt:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP-fout" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Verbindingsfout" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Time-out tijdens maken van verbinding" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Algemene fout" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Geüpload bestand kon niet opgeslagen worden in de tijdelijke map" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Bestanden kunnen niet vervangen worden tijdens een update" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Mislukt om minstens een LDAP gebruiker aan te maken" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Het is niet gelukt tenminste een LDAP gebruiker aan te maken" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Fout: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Fout: No user returned in response of LDAP server" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Minstens een LDAP Gebruiker is niet gevonden in de Database" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Gebruiker succesvol geïmporteerd" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Database niet gevonden, voer de juiste locatie in" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Kan niet schrijven naar database" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "SSL-sleutellocatie is niet geldig, voer een geldig pad in" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "SSL-certificaatlocatie is niet geldig, voer een geldig pad in" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "Het wachtwoord moet tussen de 1 en 40 tekens lang zijn" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "E-mailserver-instellingen bijgewerkt" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Databaseconfiguratie" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Vul alle velden in!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Het e-mailadres bevat geen geldige domeinnaam" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Gebruiker toevoegen" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Gebruiker '%(user)s' aangemaakt" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Bestaand account met dit e-mailadres of deze gebruikersnaam aangetroffen." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Gebruiker '%(nick)s' verwijderd" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Kan Gast gebruiker niet verwijderen" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Kan laatste systeembeheerder niet verwijderen" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "E-mail kan niet leeg zijn en moet geldig zijn" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Gebruiker '%(nick)s' bijgewerkt" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Zoeken" + #: cps/converter.py:31 msgid "not installed" msgstr "niet geïnstalleerd" @@ -530,128 +548,123 @@ msgstr "niet geïnstalleerd" msgid "Execution permissions missing" msgstr "Kan programma niet uitvoeren" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Aangepaste kolom Nr.%(column)d bestaat niet in de Calibre Database" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Geen" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Oeps! Geselecteerd boek is niet beschikbaar. Bestand bestaat niet of is niet toegankelijk" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Gebruiker mist rechten om de omslag te uploaden" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Identificatoren zijn niet hoofdlettergevoelig, overschrijf huidige identificatoren" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "De metagegevens zijn bijgewerkt" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Fout tijdens bijwerken van boek: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Bestand %(file)s geüpload" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Bron- of doelformaat ontbreekt voor conversie" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Het boek is in de wachtrij geplaatst voor conversie naar %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Er is een fout opgetreden bij het converteren van dit boek: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Geüpload boek staat mogelijk al in de bibliotheek, controleer alvorens door te gaan: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Oeps! Geselecteerd boek is niet beschikbaar. Bestand bestaat niet of is niet toegankelijk" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Gebruiker mist rechten om de omslag te uploaden" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Identificatoren zijn niet hoofdlettergevoelig, overschrijf huidige identificatoren" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s is geen geldige taal" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "De metagegevens zijn bijgewerkt" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Fout tijdens bijwerken van boek: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Geüpload boek staat mogelijk al in de bibliotheek, controleer alvorens door te gaan: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "De bestandsextensie '%(ext)s' is niet toegestaan op deze server" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "De bestandsextensie '%(ext)s' is niet toegestaan op deze server" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Het te uploaden bestand moet voorzien zijn van een extensie" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Bestand %(filename)s kon niet opgeslagen worden in de tijdelijke map" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Omslag %(file)s niet verplaatst: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Het boekformaat is verwijderd" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Het boek is verwijderd" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "U mist rechten om boeken te verwijderen" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "metagegevens bewerken" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s is geen geldig nummer, sla het over" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Gebruiker mist rechten om extra bestandsformaten te uploaden" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Kan de locatie '%(path)s' niet aanmaken (niet gemachtigd)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Kan %(file)s niet opslaan." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Bestandsformaat %(ext)s toegevoegd aan %(book)s" @@ -664,487 +677,480 @@ msgstr "Het instellen van Google Drive is niet afgerond, heractiveer Google Driv msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Het callback-domein is niet geverifieerd. Volg de stappen in de Google-ontwikkelaarsconsole om het domein te verifiëren" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s formaat niet gevonden voor boek met id: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s niet aangetroffen op Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s niet gevonden %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Versturen naar Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Deze e-mail is verstuurd via Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web - test-e-mail" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Test-e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Aan de slag met Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Registratie-e-mailadres van gebruiker: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "%(orig)s converteren naar %(format)s en versturen naar Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "%(format)s versturen naar Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s verzonden naar Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Het opgevraagde bestand kan niet worden gelezen. Ben je hiertoe gemachtigd?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Gelezen/ongelezen status kan niet aangepast worden: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Het verwijderen van de boekenmap voor boek %(id)s is mislukt, het pad heeft submappen: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Verwijderen van boek %(id)s mislukt: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, fuzzy, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Verwijder boek %(id)s alleen uit database, boek pad is ongeldig: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Kan de titel '%(src)s' niet wijzigen in '%(dest)s': %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Bestand '%(file)s' niet aangetroffen op Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Kan de titel '%(src)s' niet wijzigen in '%(dest)s': %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Boeken locatie '%(path)s' niet aangetroffen op Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "Bestaand account gevondne met dit e-mailadres" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Deze gebruikersnaam is al in gebruik" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Ongeldig E-Mail adres" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "Het wachtwoord voldoet niet aan de validatieregels" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "Pythonmodule 'advocate' is niet geïnstalleerd maar is nodig omslag uploads" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Fout bij downloaden omslag" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Onjuist omslagformaat" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Toegang tot localhost of het lokale netwerk niet toegestaant voor omslag uploaden" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Locatie aanmaken voor omslag mislukt" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Omslag-bestand is geen afbeelding of kon niet opgeslagen worden" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Alleen jpg/jpeg/png/webp/bmp bestanden worden ondersteund als omslag" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Ongeldig omslagbestand" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Alleen jpg/jpeg bestanden zijn toegestaan als omslag" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Willekeurige boeken" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "UnRar executable niet gevonden" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Fout bij het uitvoeren van UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "Kan niet schrijven naar database" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Kan programma niet uitvoeren" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Fout bij het uitvoeren van UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Voeg alle boeken toe aan de wachtrij voor het maken van een metagegevens backup" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Je kunt Calibre-Web niet vanaf de lokale computer openen om een geldige api_endpoint te krijgen voor je kobo toestel" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo Instellen" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Aanmelden bij %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "je bent ingelogd als: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Koppeling gemaakt met %(oauth)s" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Inloggen mislukt, geen gebruiker gekoppeld aan OAuth account" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Koppeling met %(oauth)s verbroken" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Ontkoppelen van %(oauth)s mislukt" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Niet gelinkt aan %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Inloggen bij GitHub mislukt." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Opvragen gebruikersinfo bij GitHub mislukt." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Inloggen bij Google mislukt." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Opvragen gebruikersinfo bij Google mislukt." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub OAuth fout, probeer het later nog eens." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Github OAuth foutmelding: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google OAuth fout, probeer het later nog eens." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google OAuth foutmelding: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} sterren" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Inloggen" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Toegangssleutel niet gevonden" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Toegangssleutel is verlopen" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Gelukt! Ga terug naar je apparaat" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Boeken" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Recent toegevoegde boeken tonen" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Populaire boeken" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Populaire boeken tonen" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Gedownloade boeken" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Gedownloade boeken tonen" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Best beoordeelde boeken" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Best beoordeelde boeken tonen" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Gelezen boeken" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Gelezen/Ongelezen boeken tonen" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Ongelezen boeken" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Ongelezen boeken tonen" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Willekeurige boeken" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Willekeurige boeken tonen" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Categorieën" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Categoriekeuze tonen" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Boekenreeksen" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Boekenreeksenkeuze tonen" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Auteurs" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Auteurkeuze tonen" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Uitgevers" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Uitgeverskeuze tonen" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Talen" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Taalkeuze tonen" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Beoordelingen" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Beoordelingen tonen" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Bestandsformaten" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Bestandsformaten tonen" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Gearchiveerde boeken" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Gearchiveerde boeken tonen" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Boekenlijst" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Boekenlijst tonen" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Zoeken" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Gepubliceerd na " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Gepubliceerd vóór " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Beoordeling <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Beoordeling >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Lees Status = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Fout tijdens het zoeken van aangepaste kolommen, start Calibre-Web opnieuw op" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Geavanceerd zoeken" @@ -1200,7 +1206,7 @@ msgstr "Het boek is verwijderd van boekenplank: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "U heeft niet voldoende rechten om een boek van deze boekenplank te verwijderen" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Boekenplank maken" @@ -1255,45 +1261,45 @@ msgstr "Een openbare boekenplank met de naam '%(title)s' bestaat al." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Een persoonlijke boekenplank met de naam '%(title)s' bestaat al." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Boekenplank: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Kan boekenplank niet openen: de boekenplank bestaat niet of is ontoegankelijk" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Taken" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Wachten" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Mislukt" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Gestart" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Voltooid" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Beëindigd" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Geannuleerd" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Onbekende status" @@ -1326,178 +1332,178 @@ msgstr "Er is een update beschikbaar. Klik op de knop hieronder om te updaten na msgid "No release information available" msgstr "Geen update-informatie beschikbaar" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Verkennen (willekeurige boeken)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Populaire boeken (meest gedownload)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Gedownloade boeken door %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Auteur: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Uitgever: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Reeks: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Beoordeling: geen" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Beoordeling: %(rating)s sterren" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Bestandsformaat: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Categorie: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Taal: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Downloads" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Beoordelingen" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Alle bestandsformaten" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Stel eerst SMTP-mail in..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Het boek is in de wachtrij geplaatst om te worden verstuurd aan %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Fout opgetreden bij het versturen van dit boek: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Stel je kindle-e-mailadres in..." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "Wacht alstublieft één minuut voor het registreren van de volgende gebruiker" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registreren" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "E-mailserver is niet geconfigureerd, neem contact op met de beheerder!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "E-mailserver is niet geconfigureerd, neem contact op met de beheerder!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Dit e-mailadres mag niet worden gebruikt voor registratie" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Er is een bevestigings-e-mail verstuurd naar je e-mailadres." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Kan de LDAP authenticatie niet activeren" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "Wacht alstublieft één minuut voor de volgende inlogpoging" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "je bent ingelogd als: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Terugvallen op login: '%(nickname)s', LDAP Server is onbereikbaar, of de gebruiker is onbekend" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Inloggen mislukt: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Verkeerde gebruikersnaam of wachtwoord" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Een nieuw wachtwoord is verzonden naar je e-mailadres" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Onbekende fout opgetreden. Probeer het later nog eens." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Geef een geldige gebruikersnaam op om je wachtwoord te herstellen" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "je bent ingelogd als: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)ss profiel" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profiel bijgewerkt" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Bestaand account met dit e-mailadres aangetroffen." @@ -1506,54 +1512,58 @@ msgstr "Bestaand account met dit e-mailadres aangetroffen." msgid "Found no valid gmail.json file with OAuth information" msgstr "Geen geldig gmail.json bestand gevonden met OAuth informatie" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s verzonden naar Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre ebook-convert %(tool)s niet gevonden" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s formaat is niet gevonden op de schijf" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "E-Book converter mislukt met een onbekende foutmelding" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-converteerder mislukt: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Omgezette bestand is niet gevonden of meer dan een bestand in map %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "E-boek-conversie mislukt: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre mislukt met foutmelding: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "E-boek-conversie mislukt: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Overzetten" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Opnieuw verbinding aan het maken met Calibre database" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "E-mail" @@ -1562,30 +1572,26 @@ msgstr "E-mail" msgid "Backing up Metadata" msgstr "metagegevens bewerken" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "%{count}s omslagminiaturen gegenereerd" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Omslag miniaturen" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "{0} serieminiaturen gegenereerd" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Cache met omslagminiaturen aan het opschonen" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Uploaden" @@ -1604,12 +1610,12 @@ msgstr "Gebruikersnaam" msgid "Email" msgstr "E-mailadres" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Kindle-e-mailadres" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Beheer" @@ -1619,8 +1625,8 @@ msgstr "Beheer" msgid "Password" msgstr "Wachtwoord" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Downloaden" @@ -1832,13 +1838,13 @@ msgid "OK" msgstr "Oké" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Annuleren" @@ -1888,16 +1894,76 @@ msgstr "Sorteren op publicatiedatum, nieuwste boeken eerst" msgid "Sort according to publishing date, oldest first" msgstr "Sorteren op publicatiedatum, oudste boeken eerst" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "beperken" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Meer van" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Boek %(index)s van %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Taal" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Uitgever" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Gepubliceerd" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Beschrijving:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Vorige" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Volgende" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Geen resultaten gevonden" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Startpagina" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Zoek in bibliotheek" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Afmelden" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Boek verwijderen" @@ -1926,99 +1992,107 @@ msgstr "Converteren naar:" msgid "Convert book" msgstr "Boek converteren" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Bezig met uploaden..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Sluiten" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Fout" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Uploaden voltooid, bezig met verwerken..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Uploadformaat" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Boektitel" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Auteur" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Labels" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Boekenreeks volgnummer" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Publicatiedatum" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Beoordeling" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Omschrijving" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identificatoren" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Identificatie type" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Identificatie waarde" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Verwijderen" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Identificator toevoegen" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Labels" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Boekenreeks volgnummer" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Beoordeling" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Omslag-url (jpg) (de omslag wordt gedownload en opgeslagen in de database)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Omslag uploaden vanaf de harde schijf" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Publicatiedatum" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Uitgever" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Taal" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ja" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nee" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Uploadformaat" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Boek inkijken na bewerking" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Metagegevens ophalen" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2026,38 +2100,32 @@ msgstr "Metagegevens ophalen" msgid "Save" msgstr "Opslaan" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Trefwoord" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Trefwoord zoeken " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Klik op de omslag om de metagegevens in het formulier te laden" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Bezig met laden..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Sluiten" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Bron" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Zoekfout!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Geen resultaten gevonden! Gebruik een ander trefwoord." @@ -2166,7 +2234,7 @@ msgid "Enter " msgstr "Identificatoren" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Weet je het zeker?" @@ -2645,74 +2713,61 @@ msgstr "Voeg toegestane/geweigerde tags toe" msgid "Add Allowed/Denied custom column values" msgstr "Voeg toegestane/geweigerde aangepaste kolomwaarden toe" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Lezen in webbrowser" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Luisteren in webbrowser" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Boek %(index)s van %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Gepubliceerd" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Markeren als ongelezen" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Markeren als gelezen" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Markeren als ongelezen" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Gelezen" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Terughalen uit archief" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Toevoegen aan archief" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Gearchiveerd" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Beschrijving:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Toevoegen aan boekenplank" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Openbaar)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Metagegevens bewerken" @@ -2785,10 +2840,6 @@ msgstr "Voer domeinnaam in" msgid "Denied Domains (Blacklist)" msgstr "Geweigerde domeinen voor registratie" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Volgende" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Open het .kobo/Kobo/Kobo eReader.conf bestand in een teksteditor en voeg toe (of bewerk):" @@ -2811,11 +2862,16 @@ msgstr "E-mailserver is niet geconfigureerd, neem contact op met de beheerder!" msgid "Create Issue" msgstr "Probleem melden" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Databaseconfiguratie" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Terug naar startpagina" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Gebruiker uitloggen" @@ -2905,7 +2961,7 @@ msgstr "Boeken gesorteerd op beoordeling" msgid "Books ordered by file formats" msgstr "Boeken gesorteerd op bestandsformaat" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Boekenplanken" @@ -2914,60 +2970,37 @@ msgstr "Boekenplanken" msgid "Books organized in shelves" msgstr "Boeken georganiseerd op boekenplanken" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Startpagina" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Navigatie aanpassen" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Zoek in bibliotheek" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Eenvoudig" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Account" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Afmelden" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Bezig met uploaden..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Fout" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Uploaden voltooid, bezig met verwerken..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Instellingen" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Deze pagina niet vernieuwen" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Verkennen" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Informatie" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Vorige" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Boekgegevens" @@ -3083,7 +3116,7 @@ msgstr "Bovenliggende map" msgid "Select" msgstr "Selecteer" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Oké" @@ -3092,36 +3125,82 @@ msgstr "Oké" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web - e-boekcatalogus" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "PDF lezer" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Kies een gebruikersnaam" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Licht" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Donker" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sepia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Annuleren" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Tekstindeling automatisch aanpassen als het zijpaneel geopend is." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "Lettertypegrootte" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Verwijderen" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Wachten" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Verticaal" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Gelezen" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Ongeldige gelezen kolom" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3303,10 +3382,6 @@ msgstr "De link vervalt na 10 minuten." msgid "Generate Series Cover Thumbnails" msgstr "Genereer serie omslagminiaturen" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Geen resultaten gevonden" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Zoekterm:" @@ -3323,11 +3398,11 @@ msgstr "Publicatiedatum van" msgid "Published Date To" msgstr "Publicatiedatum tot" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3364,11 +3439,13 @@ msgstr "Met beoordeling hoger dan" msgid "Rating Below" msgstr "Met beoordeling lager dan" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Van:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Tot:" @@ -3392,6 +3469,16 @@ msgstr "Schakel verandering van de volgorde uit" msgid "Enable Change order" msgstr "Schakel verandering van de volgorde in" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Sorteren op datum, nieuwste boeken eerst" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Sorteren op datum, oudste boeken eerst" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Delen met iedereen" @@ -3460,15 +3547,20 @@ msgstr "Voortgang" msgid "Run Time" msgstr "Looptijd" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Samenvoegen" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Acties" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Deze taak wordt geannuleerd. De voortgang van deze taak wordt opgeslagen." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Als dit een geplande taak is wordt deze opnieuw uitgevoerd tijdens de volgende geplande tijd." @@ -3476,6 +3568,10 @@ msgstr "Als dit een geplande taak is wordt deze opnieuw uitgevoerd tijdens de vo msgid "Reset user Password" msgstr "Gebruikerswachtwoord herstellen" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Taal van boeken" diff --git a/cps/translations/no/LC_MESSAGES/messages.mo b/cps/translations/no/LC_MESSAGES/messages.mo index 8e71b408..aaabb21a 100644 Binary files a/cps/translations/no/LC_MESSAGES/messages.mo and b/cps/translations/no/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/no/LC_MESSAGES/messages.po b/cps/translations/no/LC_MESSAGES/messages.po index 4007f022..bacc284d 100644 --- a/cps/translations/no/LC_MESSAGES/messages.po +++ b/cps/translations/no/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2023-01-06 11:00+0000\n" "Last-Translator: Vegard Fladby \n" "Language: no\n" @@ -16,507 +16,525 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistikk" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Server startet på nytt. Last inn siden på nytt" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Utfører avslutning av server, vennligst lukk vinduet" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Ukjent kommando" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Test e-post i kø for sending til %(email)s, sjekk Oppgaver for resultat" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Ukjent" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Admin side" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Grunnleggende konfigurasjon" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "UI-konfigurasjon" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Egendefinert kolonnenr.%(column)d finnes ikke i caliber-databasen" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Rediger brukere" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Alle" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Bruker ikke funnet" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} brukere ble slettet" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Vis alt" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Feil utformet forespørsel" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Gjestenavn kan ikke endres" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Gjesten kan ikke ha denne rollen" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Ingen administratorbruker igjen, kan ikke fjerne administratorrollen" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Verdien må være sann eller usann" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Ugyldig rolle" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Gjestene kan ikke ha denne utsikten" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Ugyldig visning" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Gjestenes lokalitet bestemmes automatisk og kan ikke angis" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Ingen gyldig lokalitet gitt" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Ikke oppgitt gyldig bokspråk" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parameter ikke funnet" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Ugyldig lesekolonne" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Ugyldig begrenset kolonne" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web-konfigurasjonen er oppdatert" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Vil du virkelig slette Kobo-tokenet?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Vil du virkelig slette dette domenet?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Vil du virkelig slette denne brukeren?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Er du sikker på at du vil slette denne hyllen?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Er du sikker på at du vil endre lokaliteter for valgte bruker(e)?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Er du sikker på at du vil endre synlige bokspråk for valgte bruker(e)?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Er du sikker på at du vil endre den valgte rollen for den(e) valgte brukeren?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Er du sikker på at du vil endre de valgte begrensningene for den(e) valgte brukeren?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Er du sikker på at du vil endre de valgte synlighetsbegrensningene for valgte bruker(e)?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Er du sikker på at du vil endre atferden for hyllesynkronisering for de(n) valgte brukeren(e)?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Er du sikker på at du vil endre plassering av Caliber-biblioteket?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calibre-Web vil søke etter oppdaterte omslag og oppdatere omslagsminiatyrbilder, kan dette ta litt tid?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Er du sikker på at du vil slette Calibre-Webs synkroniseringsdatabase for å tvinge frem en full synkronisering med Kobo Reader?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Benekte" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Tillate" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} synkroniseringsoppføringer slettet" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Merket ble ikke funnet" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Ugyldig handling" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json er ikke konfigurert for webapplikasjon" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Loggfilplasseringen er ikke gyldig, skriv inn riktig bane" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Plasseringen av tilgangsloggfilen er ikke gyldig, skriv inn riktig bane" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Angi en LDAP-leverandør, port, DN og brukerobjektidentifikator" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Vennligst skriv inn en LDAP-tjenestekonto og passord" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Angi en LDAP-tjenestekonto" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP-gruppeobjektfilter må ha én \"%s\"-formatidentifikator" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAP-gruppeobjektfilter har uovertruffen parentes" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP-brukerobjektfilter må ha én \"%s\"-formatidentifikator" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAP-brukerobjektfilter har uovertruffen parentes" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP-medlemsbrukerfilter må ha én \"%s\"-formatidentifikator" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "LDAP-medlemsbrukerfilter har uovertruffen parentes" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CA-sertifikat, sertifikat eller nøkkelplassering er ikke gyldig. Angi riktig bane" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Legg til ny bruker" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 #, fuzzy msgid "Edit Email Server Settings" msgstr "Rediger e-postserverinnstillinger" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, fuzzy, python-format msgid "Oops! Database Error: %(error)s." msgstr "Databasefeil: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Test e-post i kø for sending til %(email)s, sjekk Oppgaver for resultat" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Det oppsto en feil ved sending av test-e-posten: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Vennligst konfigurer e-postadressen din først..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 #, fuzzy msgid "Email Server Settings updated" msgstr "E-postserverinnstillinger oppdatert" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Rediger innstillinger for planlagte oppgaver" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Ugyldig starttidspunkt for spesifisert oppgave" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Ugyldig varighet for spesifisert oppgave" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Innstillinger for planlagte oppgaver er oppdatert" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 #, fuzzy msgid "Oops! An unknown error occurred. Please try again later." msgstr "En ukjent feil oppstod. Prøv igjen senere." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Innstillinger DB er ikke skrivbar" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Rediger bruker %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Passord for bruker %(user)s tilbakestilling" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Vennligst konfigurer SMTP-postinnstillingene først..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Loggfilviser" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Ber om oppdateringspakke" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Laster ned oppdateringspakken" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Pakker ut oppdateringspakken" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Erstatter filer" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Databaseforbindelser er stengt" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Stopper server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Oppdatering fullført, vennligst trykk OK og last inn siden på nytt" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Oppdatering mislyktes:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP-feil" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Tilkoblingsfeil" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Tidsavbrudd under etablering av tilkobling" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Generell feil" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "Oppdateringsfilen kunne ikke lagres i temp dir" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Filer kunne ikke erstattes under oppdatering" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Kunne ikke pakke ut minst én LDAP-bruker" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Kunne ikke opprette minst én LDAP-bruker" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Feil: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Feil: Ingen bruker ble returnert som svar fra LDAP-serveren" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Minst én LDAP-bruker ikke funnet i databasen" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Bruker ble importert" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "DB-plassering er ikke gyldig, skriv inn riktig bane" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "DB er ikke skrivbar" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Nøkkelfilplasseringen er ikke gyldig. Angi riktig bane" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Sertifikatfilplasseringen er ikke gyldig, vennligst skriv inn riktig bane" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Databaseinnstillinger oppdatert" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Databasekonfigurasjon" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 #, fuzzy msgid "Oops! Please complete all fields." msgstr "Vennligst fyll ut alle feltene!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-post er ikke fra gyldig domene" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Legg til ny bruker" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Bruker '%(user)s' opprettet" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Fant en eksisterende konto for denne e-postadressen eller navnet." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Brukeren '%(nick)s' slettet" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Kan ikke slette gjestebruker" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Ingen administratorbruker igjen, kan ikke slette bruker" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 #, fuzzy msgid "Email can't be empty and has to be a valid Email" msgstr "E-postadresse kan ikke være tom og må være en gyldig e-post" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Bruker '%(nick)s' oppdatert" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Søk" + #: cps/converter.py:31 msgid "not installed" msgstr "ikke installert" @@ -525,129 +543,124 @@ msgstr "ikke installert" msgid "Execution permissions missing" msgstr "Utførelsestillatelser mangler" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Egendefinert kolonnenr.%(column)d finnes ikke i caliber-databasen" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Ingen" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -#, fuzzy -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Oops! Den valgte boktittelen er utilgjengelig. Filen eksisterer ikke eller er ikke tilgjengelig" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Brukeren har ingen rettigheter til å laste opp cover" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Identifikatorer skiller ikke mellom store og små bokstaver, overskriver gammel identifikator" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadata ble oppdatert" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Feil ved redigering av bok: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Fil %(file)s lastet opp" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Kilde- eller målformat for konvertering mangler" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Boken ble satt i kø for konvertering til %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Det oppsto en feil ved konvertering av denne boken: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Opplastet bok finnes sannsynligvis i biblioteket, vurder å endre før du laster opp ny: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +#, fuzzy +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Oops! Den valgte boktittelen er utilgjengelig. Filen eksisterer ikke eller er ikke tilgjengelig" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Brukeren har ingen rettigheter til å laste opp cover" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Identifikatorer skiller ikke mellom store og små bokstaver, overskriver gammel identifikator" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "'%(langname)s' er ikke et gyldig språk" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadata ble oppdatert" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Feil ved redigering av bok: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Opplastet bok finnes sannsynligvis i biblioteket, vurder å endre før du laster opp ny: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Filtypen «%(ext)s» er ikke tillatt å lastes opp til denne serveren" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Filtypen «%(ext)s» er ikke tillatt å lastes opp til denne serveren" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Filen som skal lastes opp må ha en utvidelse" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Filen %(filename)s kunne ikke lagres i midlertidig dir" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Kunne ikke flytte omslagsfil %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Bokformatet er slettet" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Boken ble slettet" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Du mangler tillatelser til å slette bøker" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "redigere metadata" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s er ikke et gyldig tall, hopper over" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Brukeren har ingen rettigheter til å laste opp flere filformater" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Kunne ikke opprette banen %(path)s (Tillatelse nektet)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Kunne ikke lagre filen %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Filformat %(ext)s lagt til %(book)s" @@ -660,485 +673,478 @@ msgstr "Google Disk-konfigurasjonen er ikke fullført. Prøv å deaktivere og ak msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Tilbakeringingsdomene er ikke bekreftet. Følg fremgangsmåten for å bekrefte domenet i Googles utviklerkonsoll" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s format ikke funnet for bok-ID: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s ikke funnet på Google Disk: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s ikke funnet: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Send til E-Reader" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Denne e-posten er sendt via Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web test e-post" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Test e-post" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Kom i gang med Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Registrerings-e-post for bruker: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Konverter %(orig)s til %(format)s og send til E-Reader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Send %(format)s til E-Reader" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s sendes til E-Reader" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Den forespurte filen kunne ikke leses. Kanskje feil tillatelser?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Lesestatus kunne ikke angis: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Sletting av bokmappe for bok %(id)s mislyktes, banen har undermapper: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Sletting av bok %(id)s mislyktes: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Sletter bok %(id)s kun fra databasen, bokbanen i databasen er ikke gyldig: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Endre navn på forfatter fra: '%(src)s' til '%(dest)s' mislyktes med feil: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Fil %(file)s ikke funnet på Google Disk" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Endre navn på tittel fra: '%(src)s' til '%(dest)s' mislyktes med feil: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Finner ikke bokbane %(path)s på Google Disk" -#: cps/helper.py:682 +#: cps/helper.py:657 #, fuzzy msgid "Found an existing account for this Email address" msgstr "Fant en eksisterende konto for denne e-postadressen" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Dette brukernavnet er allerede tatt" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Ugyldig format for e-postadresse" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "Python-modulen 'advocate' er ikke installert, men er nødvendig for omslagsopplastinger" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Feil ved nedlasting av cover" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Omslagsformatfeil" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Du har ikke tilgang til localhost eller det lokale nettverket for coveropplastinger" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Kunne ikke opprette bane for dekning" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Cover-filen er ikke en gyldig bildefil, eller kunne ikke lagres" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Bare jpg/jpeg/png/webp/bmp-filer støttes som coverfile" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Ugyldig omslagsfilinnhold" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Bare jpg/jpeg-filer støttes som coverfile" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" msgstr "Dekke" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "UnRar binær fil ikke funnet" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "Feil ved kjøring av UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "DB er ikke skrivbar" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Utførelsestillatelser mangler" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Feil ved kjøring av UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Sett alle bøker i kø for sikkerhetskopiering av metadata" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Gå til Calibre-Web fra ikke-lokale vert for å få gyldig api_endpoint for kobo-enhet" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo oppsett" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Registrer deg hos %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, fuzzy, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "du er nå logget på som: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Link til %(oauth)s lyktes" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Pålogging mislyktes, ingen bruker koblet til OAuth-konto" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Fjern koblingen til %(oauth)s" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Fjern koblingen til %(oauth)s mislyktes" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Ikke koblet til %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Kunne ikke logge på med GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Kunne ikke hente brukerinformasjon fra GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Kunne ikke logge på med Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Kunne ikke hente brukerinformasjon fra Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth-feil, prøv igjen senere." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "GitHub Oauth-feil: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth-feil, prøv igjen senere." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google Oauth-feil: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Stjerner" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Logg Inn" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token ikke funnet" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token har utløpt" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Suksess! Gå tilbake til enheten din" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Bøker" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Vis nyere bøker" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Hot bøker" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Vis populære bøker" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Nedlastede bøker" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Vis nedlastede bøker" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Topprangerte bøker" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Vis best rangerte bøker" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Lese bøker" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Vis lest og ulest" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Uleste bøker" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Vis ulest" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Oppdage" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Vis tilfeldige bøker" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategorier" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Vis kategorivalg" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Serie" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Vis serieutvalg" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Forfattere" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Vis forfattervalg" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Forlag" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Vis utgivervalg" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Språk" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Vis språkvalg" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Vurderinger" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Vis vurderingsvalg" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Filformater" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Vis valg av filformater" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Arkiverte bøker" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Vis arkiverte bøker" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Bøker Liste" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Vis bokliste" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Søk" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Publisert etter " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Publisert før " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Vurdering <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Vurdering >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Lesestatus = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Feil ved søk etter egendefinerte kolonner. Start Calibre-Web på nytt" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Avansert søk" @@ -1193,7 +1199,7 @@ msgstr "Boken er fjernet fra hyllen: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Beklager at du ikke har lov til å fjerne en bok fra denne hyllen" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Lag en hylle" @@ -1246,45 +1252,45 @@ msgstr "En offentlig hylle med navnet '%(title)s' eksisterer allerede." msgid "A private shelf with the name '%(title)s' already exists." msgstr "En privat hylle med navnet '%(title)s' eksisterer allerede." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Hylle: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Feil ved åpning av hylle. Hylle finnes ikke eller er ikke tilgjengelig" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Oppgaver" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Venter" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Mislyktes" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Startet" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Ferdig" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Endte" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "avbrutt" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Ukjent status" @@ -1317,180 +1323,180 @@ msgstr "En ny oppdatering er tilgjengelig. Klikk på knappen nedenfor for å opp msgid "No release information available" msgstr "Ingen utgivelsesinformasjon tilgjengelig" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Discover (tilfeldige bøker)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Hot Books (mest nedlastede)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Lastet ned bøker av %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Forfatter: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Utgiver: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Serie: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Vurdering: Ingen" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Rangering: %(rating)s stjerner" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Filformat: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategori: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Språk: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Nedlastinger" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Rangeringsliste" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Liste over filformater" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Vennligst konfigurer SMTP-postinnstillingene først..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, fuzzy, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Boken ble satt i kø for sending til %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, fuzzy, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Oops! Det oppsto en feil ved sending av denne boken: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Vennligst oppdater profilen din med en gyldig Send til Kindle-e-postadresse." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registrere" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "E-postserveren er ikke konfigurert, kontakt administratoren din!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 #, fuzzy msgid "Oops! Email server is not configured, please contact your administrator." msgstr "E-postserveren er ikke konfigurert, kontakt administratoren din!" -#: cps/web.py:1319 +#: cps/web.py:1328 #, fuzzy msgid "Oops! Your Email is not allowed." msgstr "Din e-post kan ikke registreres" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Kan ikke aktivere LDAP-autentisering" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "du er nå logget på som: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Reservepålogging som: '%(nickname)s', LDAP-serveren er ikke tilgjengelig, eller brukeren er ukjent" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Kunne ikke logge på: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Vennligst skriv inn gyldig brukernavn for å tilbakestille passordet" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Nytt passord ble sendt til e-postadressen din" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "En ukjent feil oppstod. Prøv igjen senere." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Vennligst skriv inn gyldig brukernavn for å tilbakestille passordet" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "du er nå logget på som: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, fuzzy, python-format msgid "%(name)s's Profile" msgstr "%(name)s sin profil" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profil oppdatert" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "" @@ -1498,54 +1504,58 @@ msgstr "" msgid "Found no valid gmail.json file with OAuth information" msgstr "Fant ingen gyldig gmail.json-fil med OAuth-informasjon" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s sendes til E-Reader" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Caliber ebook-convert %(tool)s ble ikke funnet" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s format ikke funnet på disken" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Ebook-konvertering mislyktes med ukjent feil" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-konvertering mislyktes: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Konvertert fil ikke funnet eller mer enn én fil i mappen %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ebook-konvertering mislyktes: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Caliber mislyktes med feil: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ebook-konvertering mislyktes: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Konvertere" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Kobler til Caliber-databasen på nytt" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "E-post" @@ -1553,30 +1563,26 @@ msgstr "E-post" msgid "Backing up Metadata" msgstr "Sikkerhetskopierer metadata" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "Genererte %(count)s forsideminiatyrbilder" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Forsideminiatyrbilder" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "Genererte {0} serieminiatyrbilder" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Tømmer cache for miniatyrbilde for omslag" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Laste opp" @@ -1596,12 +1602,12 @@ msgstr "Brukernavn" msgid "Email" msgstr "E-post" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Send til E-Reader E-postadresse" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Admin" @@ -1611,8 +1617,8 @@ msgstr "Admin" msgid "Password" msgstr "Passord" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "nedlasting" @@ -1829,13 +1835,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Avbryt" @@ -1885,16 +1891,79 @@ msgstr "Sorter etter publiseringsdato, nyeste først" msgid "Sort according to publishing date, oldest first" msgstr "Sorter etter publiseringsdato, eldste først" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "redusere" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Mer av" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Sletting av bok %(index)s mislyktes: %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Språk" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Forlegger" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +#, fuzzy +msgid "Published" +msgstr "Forlegger" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +#, fuzzy +msgid "Description:" +msgstr "Beskrivelse" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +#, fuzzy +msgid "No Results Found" +msgstr "Merket ble ikke funnet" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Slett bok" @@ -1923,99 +1992,109 @@ msgstr "Konvertere til:" msgid "Convert book" msgstr "Konverter bok" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +#, fuzzy +msgid "Uploading..." +msgstr "Laster inn..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Lukk" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#, fuzzy +msgid "Error" +msgstr "Serverport" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "" + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Last opp format" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Boktittel" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Forfatter" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Tagger" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Serie-ID" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Publiseringsdato" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Vurdering" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Beskrivelse" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identifikatorer" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Identifikatortype" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Identifikatorverdi" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Ta bort" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Legg til identifikator" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Tagger" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Serie-ID" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Vurdering" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Hent omslag fra URL (JPEG - Bilde vil bli lastet ned og lagret i databasen)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Last opp cover fra lokal disk" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Publiseringsdato" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Forlegger" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Språk" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ja" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nei" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Last opp format" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Se bok på Lagre" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Hent metadata" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2023,37 +2102,31 @@ msgstr "Hent metadata" msgid "Save" msgstr "Lagre" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Nøkkelord" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Søk nøkkelord" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Klikk på omslaget for å laste metadata til skjemaet" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Laster inn..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Lukk" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Kilde" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Søkefeil!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Ingen resultater! Prøv et annet søkeord." @@ -2160,7 +2233,7 @@ msgid "Enter " msgstr "Tast inn " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Er du virkelig sikker?" @@ -2641,77 +2714,62 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Sletting av bok %(index)s mislyktes: %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -#, fuzzy -msgid "Published" -msgstr "Forlegger" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" msgstr "" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 #, fuzzy msgid "Read" msgstr "redusere" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Arkiverte bøker" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -#, fuzzy -msgid "Description:" -msgstr "Beskrivelse" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 #, fuzzy msgid "Add to shelf" msgstr "Rediger en hylle" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 #, fuzzy msgid "Edit Metadata" msgstr "redigere metadata" @@ -2785,10 +2843,6 @@ msgstr "Skriv inn kommentarer" msgid "Denied Domains (Blacklist)" msgstr "" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2810,11 +2864,16 @@ msgstr "E-postserveren er ikke konfigurert, kontakt administratoren din!" msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Databasekonfigurasjon" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2910,7 +2969,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2919,65 +2978,40 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "" - #: cps/templates/layout.html:32 #, fuzzy msgid "Toggle Navigation" msgstr "Loggfilkonfigurasjon" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Enkel" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -#, fuzzy -msgid "Uploading..." -msgstr "Laster inn..." - -#: cps/templates/layout.html:78 -#, fuzzy -msgid "Error" -msgstr "Serverport" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "" - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 #, fuzzy msgid "Settings" msgstr "Vurderinger" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 #, fuzzy msgid "Please do not refresh the page" msgstr "Oppdaterer, vennligst ikke last inn denne siden på nytt" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 #, fuzzy msgid "Book Details" msgstr "Detaljer" @@ -3099,7 +3133,7 @@ msgstr "" msgid "Select" msgstr "Slett" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "" @@ -3107,35 +3141,81 @@ msgstr "" msgid "Calibre-Web eBook Catalog" msgstr "" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Bruk e-post som brukernavn" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 #, fuzzy msgid "Light" msgstr "Nattlig" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Slett" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Venter" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "E-post" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "redusere" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Ugyldig lesekolonne" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "" @@ -3320,11 +3400,6 @@ msgstr "" msgid "Generate Series Cover Thumbnails" msgstr "Generer serieomslagsminiatyrbilder" -#: cps/templates/search.html:6 -#, fuzzy -msgid "No Results Found" -msgstr "Merket ble ikke funnet" - #: cps/templates/search.html:7 #, fuzzy msgid "Search Term:" @@ -3344,11 +3419,11 @@ msgstr "Publisert etter " msgid "Published Date To" msgstr "Publisert etter " -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3386,11 +3461,13 @@ msgstr "Vurdering: Ingen" msgid "Rating Below" msgstr "" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3414,6 +3491,16 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Sorter etter bokdato, nyeste først" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Sorter etter bokdato, eldste først" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "" @@ -3488,16 +3575,21 @@ msgstr "" msgid "Run Time" msgstr "" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Slå sammen" + +#: cps/templates/tasks.html:21 #, fuzzy msgid "Actions" msgstr "Vurderinger" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3505,6 +3597,10 @@ msgstr "" msgid "Reset user Password" msgstr "" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 #, fuzzy msgid "Language of Books" diff --git a/cps/translations/pl/LC_MESSAGES/messages.mo b/cps/translations/pl/LC_MESSAGES/messages.mo index cd4d30c9..5770f962 100644 Binary files a/cps/translations/pl/LC_MESSAGES/messages.mo and b/cps/translations/pl/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/pl/LC_MESSAGES/messages.po b/cps/translations/pl/LC_MESSAGES/messages.po index 3ddab9b2..50af4abc 100644 --- a/cps/translations/pl/LC_MESSAGES/messages.po +++ b/cps/translations/pl/LC_MESSAGES/messages.po @@ -8,7 +8,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre Web - polski (POT: 2021-06-12 08:52)\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2021-06-12 15:35+0200\n" "Last-Translator: Radosław Kierznowski \n" "Language: pl\n" @@ -17,512 +17,530 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statystyki" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Serwer uruchomiony ponownie, proszę odświeżyć stronę" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Wykonano wyłączenie serwera, proszę zamknąć okno" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Nieznane polecenie" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Testowy e-mail czeka w kolejce do wysłania do %(email)s, sprawdź zadania, aby uzyskać wynik" # ??? -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Nieznany" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Panel administratora" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Konfiguracja podstawowa" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Konfiguracja Interfejsu" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Niestandardowa kolumna No.%(column)d nie istnieje w bazie calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Edytuj użytkowników" # ??? -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Wszystko" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Nie znaleziono użytkownika" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} użytkowników usuniętych pomyślnie" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Pokaż wszystkie" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Nieprawidłowo sformułowane żądanie" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Nazwa gościa nie może być zmieniona" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Gość nie może pełnić tej roli" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Nie można odebrać praw administratora. Brak na serwerze innego konta z prawami administratora" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Wartość musi być prawdziwa lub fałszywa" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Nieprawidłowa rola" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Gość nie może tego zobaczyć" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Nieprawidłowy widok" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Lokalizacja gościa jest określana automatycznie i nie można jej ustawić" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Nie podano prawidłowej lokalizacji" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Nie podano obowiązującego języka książki" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Nie znaleziono parametru" -#: cps/admin.py:572 +#: cps/admin.py:578 #, fuzzy msgid "Invalid Read Column" msgstr "Nieprawidłowa kolumna odczytu" -#: cps/admin.py:578 +#: cps/admin.py:584 #, fuzzy msgid "Invalid Restricted Column" msgstr "Nieprawidłowa kolumna z ograniczeniami" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Konfiguracja Calibre-Web została zaktualizowana" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Czy na pewno chcesz usunąć Token Kobo?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Czy naprawdę chcesz usunąć tę domenę?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Czy naprawdę chcesz usunąć tego użytkownika?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Czy na pewno chcesz usunąć półkę?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Czy na pewno chcesz zmienić ustawienia lokalne wybranego użytkownika(ów)?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Czy na pewno chcesz zmienić widoczne języki książek dla wybranego użytkownika (użytkowników)?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Czy na pewno chcesz zmienić wybraną rolę dla wybranego użytkownika (użytkowników)?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Czy na pewno chcesz zmienić wybrane ograniczenia dla wybranego użytkownika(ów)?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Czy na pewno chcesz zmienić wybrane ograniczenia widoczności dla wybranego użytkownika(ów)?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Czy na pewno chcesz zmienić zachowanie synchronizacji półek dla wybranego użytkownika(ów)?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Czy na pewno chcesz zmienić lokalizację biblioteki Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Zabroń" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Zezwalaj" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 #, fuzzy msgid "Tag not found" msgstr "Nie znaleziono znacznika" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Nieprawidłowe działanie" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json nie został skonfigurowany dla aplikacji webowej" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Lokalizacja pliku dziennika jest nieprawidłowa, wprowadź poprawną ścieżkę" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Lokalizacja pliku dziennika dostępu jest nieprawidłowa, wprowadź poprawną ścieżkę" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Wprowadź dostawcę LDAP, port, nazwę wyróżniającą i identyfikator obiektu użytkownika" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Proszę wprowadzić konto i hasło usługi LDAP" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Proszę wprowadzić konto usługi LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Filtr obiektów grupy LDAP musi mieć jeden identyfikator formatu \"% s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Filtr obiektów grupy LDAP ma niedopasowany nawias" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Filtr obiektów użytkownika LDAP musi mieć jeden identyfikator formatu \"% s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Filtr obiektów użytkownika LDAP ma niedopasowany nawias" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Filtr użytkownika członka LDAP musi mieć jedno \"%s\" identyfikator formatu" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Filtr użytkownika członka LDAP ma niedopasowane nawiasy" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "Główny urząd certyfikatu LDAP, Certyfikat lub Lokalizacja Klucza nie jest prawidłowa, Proszę wprowadzić poprawną ścieżkę" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Dodaj nowego użytkownika" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Zmień ustawienia SMTP" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Błąd bazy danych: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, fuzzy, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Testowy e-mail czeka w kolejce do wysłania do %(email)s, sprawdź zadania, aby uzyskać wynik" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Wystąpił błąd podczas wysyłania e-maila testowego: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Najpierw skonfiguruj swój adres e-mail..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Zaktualizowano ustawienia serwera poczty e-mail" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Wystąpił nieznany błąd. Spróbuj ponownie później." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Edytuj użytkownika %(nick)s" # ??? -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Zrestartowano hasło użytkownika %(user)s" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Proszę najpierw skonfigurować ustawienia SMTP poczty e-mail..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Przeglądanie dziennika" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Żądanie o pakiet aktualizacji" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Pobieranie pakietu aktualizacji" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Rozpakowywanie pakietu aktualizacji" # ??? -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Zastępowanie plików" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Połączenia z bazą danych zostały zakończone" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Zatrzymywanie serwera" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Aktualizacja zakończona, proszę nacisnąć OK i odświeżyć stronę" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Aktualizacja nieudana:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Błąd HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Błąd połączenia" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Przekroczono limit czasu podczas nawiązywania połączenia" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Błąd ogólny" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Plik aktualizacji nie mógł zostać zapisany w katalogu tymczasowym" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Błąd przy tworzeniu przynajmniej jednego użytkownika LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Błąd przy tworzeniu przynajmniej jednego użytkownika LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Błąd: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Błąd. LDAP nie zwrócił żadnego użytkownika" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Przynajmniej jeden użytkownik LDAP nie został znaleziony w bazie danych" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Użytkownik pomyślnie zaimportowany" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Lokalizacja bazy danych jest nieprawidłowa, wprowadź poprawną ścieżkę" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Baza danych nie jest zapisywalna" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Lokalizacja pliku klucza jest nieprawidłowa, wprowadź poprawną ścieżkę" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Lokalizacja pliku certyfikatu jest nieprawidłowa, wprowadź poprawną ścieżkę" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Zaktualizowano ustawienia serwera poczty e-mail" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Konfiguracja bazy danych" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Proszę wypełnić wszystkie pola!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-mail nie pochodzi z prawidłowej domeny" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Dodaj nowego użytkownika" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Użytkownik '%(user)s' został utworzony" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Znaleziono istniejące konto dla tego adresu e-mail lub nazwy." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Użytkownik '%(nick)s' został usunięty" -#: cps/admin.py:1982 +#: cps/admin.py:2005 #, fuzzy msgid "Can't delete Guest User" msgstr "Nie można usunąć użytkownika gościa" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Nie można usunąć użytkownika. Brak na serwerze innego konta z prawami administratora" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Użytkownik '%(nick)s' został zaktualizowany" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Szukaj" + #: cps/converter.py:31 msgid "not installed" msgstr "nie zainstalowane" @@ -531,128 +549,123 @@ msgstr "nie zainstalowane" msgid "Execution permissions missing" msgstr "Brak uprawnienia do wykonywania pliku" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Niestandardowa kolumna No.%(column)d nie istnieje w bazie calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Brak" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Błąd otwierania e-booka. Plik nie istnieje lub jest niedostępny" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "W identyfikatorach nie jest rozróżniana wielkość liter, nadpisywanie starego identyfikatora" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadane zostały pomyślnie zaktualizowane" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Wysłano plik %(file)s" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Brak formatu źródłowego lub docelowego do konwersji" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Książka została pomyślnie umieszczona w zadaniach do konwersji %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Podczas konwersji książki wystąpił błąd: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Wysłana książka prawdopodobnie istnieje w bibliotece, rozważ zmianę przed przesłaniem nowej: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Błąd otwierania e-booka. Plik nie istnieje lub jest niedostępny" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "W identyfikatorach nie jest rozróżniana wielkość liter, nadpisywanie starego identyfikatora" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s nie jest prawidłowym językiem" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadane zostały pomyślnie zaktualizowane" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Wysłana książka prawdopodobnie istnieje w bibliotece, rozważ zmianę przed przesłaniem nowej: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Rozszerzenie pliku '%(ext)s' nie jest dozwolone do wysłania na ten serwer" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Rozszerzenie pliku '%(ext)s' nie jest dozwolone do wysłania na ten serwer" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Plik do wysłania musi mieć rozszerzenie" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Nie można zapisać pliku %(filename)s w katalogu tymczasowym" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Nie udało się przenieść pliku okładki %(file)s:%(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Plik książki w wybranym formacie został usunięty" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Książka została usunięta" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "edytuj metadane" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s nie jest poprawną liczbą, pomijanie" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Nie udało się utworzyć łącza %(path)s (Odmowa dostępu)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Nie można zapisać pliku %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Format pliku %(ext)s dodany do %(book)s" @@ -665,491 +678,484 @@ msgstr "Konfiguracja Google Drive nie została zakończona, spróbuj dezaktywowa msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Zwrotna domena nie jest zweryfikowana, proszę zweryfikowania domenę w konsoli deweloperskiej google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "Nie znaleziono formatu %(format)s dla id książki: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "Nie znaleziono %(format)s na Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s nie znaleziono: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Wyślij do Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Ten e-mail został wysłany za pośrednictwem Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Testowy e-mail Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Testowy e-mail" # ??? -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Zacznij korzystać z Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Rejestracja e-mail dla użytkownika: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Konwertuj %(orig)s do %(format)s i wyślij do Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Wyślij %(format)s do Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Wyślij do Kindle" # ??? -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Żądany plik nie mógł zostać odczytany. Może brakuje uprawnień?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Usuwanie folderu książki dla książki %(id)s nie powiodło się, ścieżka ma podfoldery: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Usuwanie książki %(id)s zakończyło się błędem: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, fuzzy, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Usuwanie książki %(id)s, ścieżka książki jest niepoprawna: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Zmiana nazwy tytułu z: „%(src)s” na „%(dest)s” zakończyła się błędem: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Nie znaleziono pliku %(file)s na Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Zmiana nazwy tytułu z: „%(src)s” na „%(dest)s” zakończyła się błędem: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Nie znaleziono ścieżki do książki %(path)s na Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Nazwa użytkownika jest już zajęta" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Nieprawidłowy format adresu e-mail" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Błąd przy pobieraniu okładki" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Błędny format okładki" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Nie udało się utworzyć ścieżki dla okładki" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Plik okładki nie jest poprawnym plikiem obrazu lub nie mógł zostać zapisany" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Jako plik okładki obsługiwane są tylko pliki jpg/jpeg/png/webp/bmp" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Jako plik okładki dopuszczalne są jedynie pliki jpg/jpeg" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Odkrywaj" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Plik wykonywalny programu unrar nie znaleziony" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Błąd przy wykonywaniu unrar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "Baza danych nie jest zapisywalna" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Brak uprawnienia do wykonywania pliku" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Błąd przy wykonywaniu unrar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Aby uzyskać prawidłowy api_endpoint dla urządzenia Kobo, należy skorzystać z dostępu do calibre-web spoza localhost" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Konfiguracja Kobo" # ??? -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Zarejestruj się %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "zalogowałeś się jako: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Łączenie z %(oauth)s zakończono sukcesem" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Błąd logowania, użytkownik niepołączony z kontem OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Rozłączanie z %(oauth)s zakończono sukcesem" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Rozłączanie z %(oauth)s zakończono porażką" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Nie połączono z %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Nie udało się zalogować za pomocą GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Nie udało się pobrać informacji o użytkowniku z GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Nie udało się zalogować do Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Nie udało się pobrać informacji o użytkowniku z Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Błąd GitHub Oauth, proszę spróbować później." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Błąd GitHub Oauth: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Błąd Google Oauth, proszę spróbować później." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Błąd Google Oauth: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Gwiazdek" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Zaloguj się" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Nie znaleziono tokenu" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token wygasł" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Powodzenie! Wróć do swojego urządzenia" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Książki" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Pokaż menu ostatnio dodanych książek" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Najpopularniejsze" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Pokaż menu najpopularniejszych książek" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Pobrane książki" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Pokaż pobrane książki" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Najwyżej ocenione" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Pokaż menu najwyżej ocenionych książek" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Przeczytane" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Pokaż menu przeczytane i nieprzeczytane" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Nieprzeczytane" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Pokaż nieprzeczytane" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Odkrywaj" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Pokazuj losowe książki" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategorie" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Pokaż menu wyboru kategorii" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Cykle" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Pokaż menu wyboru cyklu" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autorzy" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Pokaż menu wyboru autora" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Wydawcy" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Pokaż menu wyboru wydawcy" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Języki" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Pokaż menu wyboru języka" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Oceny" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Pokaż menu listy ocen" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formaty plików" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Pokaż menu formatu plików" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Zarchiwizowane książki" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Pokaż zarchiwizowane książki" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Lista książek" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Pokaż listę książek" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Szukaj" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Opublikowane po " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Opublikowane przed " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Ocena <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Ocena >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Status przeczytania = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 #, fuzzy msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Błąd podczas wyszukiwania kolumn niestandardowych, proszę zrestartować Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Wyszukiwanie" @@ -1205,7 +1211,7 @@ msgstr "Książka została usunięta z półki: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Utwórz półkę" @@ -1260,45 +1266,45 @@ msgstr "Publiczna półka o nazwie '%(title)s' już istnieje." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Prywatna półka o nazwie '%(title)s' już istnieje." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Półka: „%(name)s”" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Błąd otwierania półki. Półka nie istnieje lub jest niedostępna" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Zadania" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Oczekiwanie" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Nieudane" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Rozpoczynanie" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Zakończone" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Ststus nieznany" @@ -1331,178 +1337,178 @@ msgstr "Dostępna jest nowa aktualizacja. Kliknij przycisk poniżej, aby zaktual msgid "No release information available" msgstr "Brak dostępnych informacji o wersji" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Odkrywaj (losowe książki)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Najpopularniejsze książki (najczęściej pobierane)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Książki pobrane przez %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Autor: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Wydawca: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Cykl: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Ocena: %(rating)s gwiazdek" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Format pliku: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategoria: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Język: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "DLS" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Lista z ocenami" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Lista formatów" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Proszę najpierw skonfigurować ustawienia SMTP poczty e-mail..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Książka została umieszczona w kolejce do wysłania do %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Wystąpił błąd podczas wysyłania tej książki: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Najpierw skonfiguruj adres e-mail Kindle..." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Zarejestruj się" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Serwer e-mail nie jest skonfigurowany, skontaktuj się z administratorem!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Serwer e-mail nie jest skonfigurowany, skontaktuj się z administratorem!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Twój e-mail nie może się zarejestrować" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Wiadomość e-mail z potwierdzeniem została wysłana na Twoje konto e-mail." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Nie można aktywować uwierzytelniania LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "zalogowałeś się jako: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Fallback Login as: %(nickname)s, LDAP Server not reachable, or user not known" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Nie można zalogować: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Błędna nazwa użytkownika lub hasło" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Nowe hasło zostało wysłane na Twój adres e-mail" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Wystąpił nieznany błąd. Spróbuj ponownie później." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Wprowadź prawidłową nazwę użytkownika, aby zresetować hasło" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "zalogowałeś się jako: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Profil użytkownika %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Zaktualizowano profil" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Znaleziono istniejące konto dla tego adresu e-mail" @@ -1510,54 +1516,58 @@ msgstr "Znaleziono istniejące konto dla tego adresu e-mail" msgid "Found no valid gmail.json file with OAuth information" msgstr "Nie znaleziono poprawnego pliku gmail.json z informacjami OAuth" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Wyślij do Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Nie znaleziono narzędzia calibre %(tool)s do konwertowania" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "Nie znaleziono na dysku formatu %(format)s" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Konwertowanie ebooka zakończyło się niepowodzeniem z nieznanego powodu" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-converter spowodowało błąd: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Konwertowany plik nie został znaleziony, lub więcej niż jeden plik w folderze %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Konwertowanie nie powiodło się: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, fuzzy, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre nie powiodło się z błędem: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Konwertowanie nie powiodło się: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1566,30 +1576,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "edytuj metadane" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Wysyłanie" @@ -1608,13 +1614,13 @@ msgstr "Nazwa użytkownika" msgid "Email" msgstr "E-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Adres e-mail dla wysyłania do Kindle" # ??? -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Panel administratora" @@ -1625,8 +1631,8 @@ msgid "Password" msgstr "Hasło" # ??? -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Pobieranie" @@ -1838,13 +1844,13 @@ msgid "OK" msgstr "OK" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Anuluj" @@ -1894,16 +1900,77 @@ msgstr "Sortuj według daty publikacji, najnowsze jako pierwsze" msgid "Sort according to publishing date, oldest first" msgstr "Sortuj według daty publikacji, najstarsze jako pierwsze" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "zwiń" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Więcej według" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Język" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Wydawca" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Data publikacji" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Opis:" + +# ??? +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Poprzedni" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Następne" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Brak wyników" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Główne menu" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Przeszukaj bibliotekę" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Wyloguj się" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Usuń książkę" @@ -1933,99 +2000,108 @@ msgstr "Konwertuj na:" msgid "Convert book" msgstr "Konwertuj książkę" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Wysyłanie…" + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Zamknij" + +# ??? +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Błąd" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Wysyłanie zakończone, przetwarzanie, proszę czekać…" + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Wyślij format" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Tytuł książki" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Etykiety" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID cyklu" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Data publikacji" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Ocena" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Opis" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identyfikatory" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Rodzaj identyfikatora" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Wartość identyfikatora" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Usuń" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Dodaj identyfikator" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Etykiety" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID cyklu" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Ocena" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Pobierz okładkę z linku (JPEG - obraz zostanie pobrany i zapisany w bazie)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Wyślij okładkę z dysku lokalnego" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Data publikacji" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Wydawca" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Język" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Tak" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nie" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Wyślij format" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Po zapisaniu wyświetl szczegóły książki" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Uzyskaj metadane" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2033,38 +2109,32 @@ msgstr "Uzyskaj metadane" msgid "Save" msgstr "Zapisz" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Słowo kluczowe" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Szukaj słowa kluczowego " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Kliknij okładkę, aby załadować metadane do formularza" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Ładowanie..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Zamknij" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Źródło" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Błąd wyszukiwania!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nie znaleziono! Spróbuj użyć innego słowa kluczowego." @@ -2175,7 +2245,7 @@ msgid "Enter " msgstr "Identyfikatory" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Czy jesteś pewny?" @@ -2657,74 +2727,61 @@ msgstr "Dodaj dozwolone/zabronione etykiety" msgid "Add Allowed/Denied custom column values" msgstr "Dodaj dozwolone/zabronione wartości własnych kolumn" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Czytaj w przeglądarce" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Słuchaj w przeglądarce" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Data publikacji" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Oznacz jako nieprzeczytane" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Oznacz jako przeczytane" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Oznacz jako nieprzeczytane" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Przeczytana" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Przywróć z archiwum" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Dodaj do archiwum" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Zarchiwizowane" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Opis:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Dodaj do półki" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(publiczna)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Edytuj metadane" @@ -2797,10 +2854,6 @@ msgstr "Podaj nazwę domeny" msgid "Denied Domains (Blacklist)" msgstr "Domeny zabronione (czarna lista)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Następne" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Otwórz plik .kobo/Kobo/Kobo eReader.conf w edytorze tekstu i dodaj (lub edytuj):" @@ -2823,12 +2876,17 @@ msgstr "Instancja Calibre-Web jest nieskonfigurowana, proszę skontaktować się msgid "Create Issue" msgstr "Zgłoś błąd" +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Konfiguracja bazy danych" + # ??? -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Powrót do głównego menu" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 #, fuzzy msgid "Logout User" msgstr "Wyloguj użytkownika" @@ -2919,7 +2977,7 @@ msgstr "Książki sortowane według oceny" msgid "Books ordered by file formats" msgstr "Ksiązki sortowane według formatu" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Półki" @@ -2928,62 +2986,37 @@ msgstr "Półki" msgid "Books organized in shelves" msgstr "Książki ułożone na półkach" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Główne menu" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Przełącz nawigację" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Przeszukaj bibliotekę" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Proste" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Konto" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Wyloguj się" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Wysyłanie…" - -# ??? -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Błąd" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Wysyłanie zakończone, przetwarzanie, proszę czekać…" - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Ustawienia" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Proszę nie odświeżać strony" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Przeglądaj" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Informacje" -# ??? -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Poprzedni" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Szczegóły książki" @@ -3100,7 +3133,7 @@ msgstr "Główny katalog" msgid "Select" msgstr "Wybierz" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "OK" @@ -3108,36 +3141,83 @@ msgstr "OK" msgid "Calibre-Web eBook Catalog" msgstr "Katalog e-booków Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "Czytnik PDF" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Wybierz nazwę użytkownika" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Jasny" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Ciemny" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Wróć" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Przepływ tekstu, gdy paski boczne są otwarte." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +# ??? +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Usuń" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Oczekiwanie" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Pionowo" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Przeczytana" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Nieprawidłowa kolumna odczytu" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3319,10 +3399,6 @@ msgstr "Link wygaśnie po 10 minutach." msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Brak wyników" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Wyszukiwano:" @@ -3339,11 +3415,11 @@ msgstr "Data publikacji od" msgid "Published Date To" msgstr "Data publikacji do" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3379,11 +3455,13 @@ msgstr "Ocena większa niż" msgid "Rating Below" msgstr "Ocena mniejsza niż" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Od:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Do:" @@ -3407,6 +3485,16 @@ msgstr "Wyłączenie Zlecenie zmiany" msgid "Enable Change order" msgstr "Włącz polecenie zmiany" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Sortuj książki według daty, najnowsze jako pierwsze" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Sortuj książki według daty, najstarsze jako pierwsze" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Współdziel z wszystkimi" @@ -3476,15 +3564,20 @@ msgstr "Postęp" msgid "Run Time" msgstr "Czas wykonania" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Połącz" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3492,6 +3585,10 @@ msgstr "" msgid "Reset user Password" msgstr "Zresetuj hasło użytkownika" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Pokaż książki w języku" diff --git a/cps/translations/pt/LC_MESSAGES/messages.mo b/cps/translations/pt/LC_MESSAGES/messages.mo index b3cf448a..7f24fcef 100644 Binary files a/cps/translations/pt/LC_MESSAGES/messages.mo and b/cps/translations/pt/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/pt/LC_MESSAGES/messages.po b/cps/translations/pt/LC_MESSAGES/messages.po index 6b1e05b5..d9eb56dc 100644 --- a/cps/translations/pt/LC_MESSAGES/messages.po +++ b/cps/translations/pt/LC_MESSAGES/messages.po @@ -4,7 +4,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2023-07-25 11:30+0100\n" "Last-Translator: horus68 \n" "Language: pt\n" @@ -13,501 +13,519 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Estatísticas" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Servidor reiniciado, por favor, refresque a página" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "A encerrar o servidor, por favor, feche a janela" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "Sucesso! Base de dados religada" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Comando desconhecido" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Sucesso! Livros enviados para lista de espera para cópia de segurança de metadados. Por favor, verifique o resultado em Tarefas" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Desconhecido" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Página de administração" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Configuração básica" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Configuração de IU" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "A coluna personalizada No.%(column)d não existe na base de dados do Calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Editar utilizadores" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Tudo" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Utilizador não encontrado" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} utilizadores eliminados com sucesso" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Mostrar tudo" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Pedido mal construído" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Nome do convidado não pode ser alterado" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Convidado não pode ter esta função" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Nenhum utilizador administrador restante, impossível remover a função de administrador" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Valor tem de ser verdadeiro ou falso" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Função inválida" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "O convidado não podr ter esta vista" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Vista inválida" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "O idioma do convidado é detetado automaticamente e não pode ser alterado" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Nenhum idioma válido fornecido" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Não foi indicado um idioma de livro válido" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parâmetro não encontrado" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Coluna Lido é inválida" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Coluna Restrito é inválida" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Configuração do Calibre-Web atualizada" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Quer realmente apagar a Kobo Token?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Quer realmente eliminar este domínio?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Quer realmente apagar este utilizador?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Tem a certeza de que quer apagar essa estante?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Tem a certeza de que quer alterar o idioma dos utilizadores selecionados?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Tem a certeza de que quer alterar os idiomas de livros visíveis para os utilizadores selecionados?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Tem a certeza de que quer alterar a função selecionada para os utilizadores selecionados?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Tem a certeza de que quer alterar as restrições selecionadas para os utilizadores selecionados?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Tem a certeza de de que quer alterar as restrições de visibilidade selecionadas para os utilizadores selecionados?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Tem a certeza de que quer alterar o comportamento de sincronização da estante para os utilizadores selecionados?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Tem a certeza de que quer alterar a localização da biblioteca Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "O Calibre-Web irá procurar por capas atualizadas e atualizará as miniaturas das capas. Isto poderá demorar um pouco" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Tem a certeza de que deseja apagar a base de dados de sincronização do Calibre-Web para forçar uma sincronização completa com seu Kobo Reader?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Negar" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Permitir" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} entradas de sincronização eliminadas" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Etiqueta não encontrada" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Ação inválida" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json não está configurado para aplicação Web" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "A localização do ficheiro de historial não é válida. Por favor, digite um caminho correto" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "A localização do ficheiro de historial não é válida. Por favor, digite um caminho correto" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Digite um fornecedor LDAP, porta, DN e identificador de objeto do utilizador" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Por favor, digite uma conta de serviço LDAP e senha" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Por favor, digite uma conta de serviço LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "O filtro de objeto de grupo LDAP precisa de ter um identificador de formato \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Filtro de objeto de grupo LDAP tem parênteses não coincidentes" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "O filtro de objeto de utilizador LDAP precisa de ter um identificador de formato \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Filtro de objeto de utilizador LDAP tem parênteses não coincidentes" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "O filtro de utilizador membro do LDAP precisa ter um identificador de formato \"%s\"" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Filtro de utilizador de membro LDAP tem parênteses incomparáveis" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP Certificado CA: Localização inválida de certificado ou chave. Por favor, digite um caminho correto" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Adicionar utilizador" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Editar configurações do servidor de email" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "Sucesso! Conta Gmail verificada" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Erro de base de dados: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Email de teste enviado para lista de espera para envio para %(email)s. Por favor, verifique o resultado em Tarefas" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Ocorreu um erro ao enviar o email de teste: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Por favor, primeiro configure seu endereço de email..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Atualização das configurações do servidor de email" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Editar configurações de tarefas agendadas" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Hora de início inválida para a tarefa especificada" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Duração inválida para a tarefa especificada" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Foram atualizadas as configurações de tarefas agendadas" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Ocorreu um erro desconhecido. Por favor, tente novamente mais tarde." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Configuaração da DB não é gravável" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Editar utilizador %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Sucesso! Senha do utilizador %(user)s redefinida" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Por favor, configure primeiro as configurações de correio SMTP..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Visualizador do historial" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "A solicitar pacote de atualização" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "A descarregar pacote de atualização" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "A descomprimir pacote de atualização" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "A substituir ficheiros" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "As ligações à base de dados estão fechadas" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "A parar o servidor" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Atualização concluída, pressione OK e refresque a página" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Atualização falhou:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Erro HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Erro de ligação" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Tempo limite ao estabelecer a ligação" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Erro geral" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "Ficheiro de atualização não pode ser guardado na pasta temporária" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Não foi possível substituir ficheiros durante a atualização" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Falha ao extrair pelo menos um utilizador LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Falha ao criar pelo menos um utilizador LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Erro: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Erro: Nenhum utilizador devolvido na resposta do servidor LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "No mínimo um utilizador LDAP não encontrado no base de dados" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} utilizador importado com sucesso" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "A localização da base de dados não é válida. Por favor, digite o caminho correto" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "DB não é gravável" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Localização do ficheiro de chaves não é válida. Por favor, digite o caminho correto" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Localização do ficheiro de certificados não é válida. Por favor, digite o caminho correto" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "A dimensão da senha deve estar entre 1 e 40" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Configurações da base de dados atualizadas" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Configuração da base de dados" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Por favor, preencha todos os campos!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "O email não é de um domínio válido" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Adicionar novo utilizador" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Criado utilizador '%(user)s'" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Já existe uma conta para este endereço de email ou nome." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Utilizador '%(nick)s' eliminado" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Impossível eliminar convidado" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Nenhum utilizador administrador restante, não é possível eliminar o utilizador" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "O email não pode estar vazio e tem de ser válido" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Utilizador '%(nick)s' atualizado" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Pesquisar" + #: cps/converter.py:31 msgid "not installed" msgstr "não instalado" @@ -516,128 +534,123 @@ msgstr "não instalado" msgid "Execution permissions missing" msgstr "Falta de permissões de execução" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "A coluna personalizada No.%(column)d não existe na base de dados do Calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Nenhum" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Oops! O Livro selecionado não está disponível. O ficheiro não existe ou não está acessível" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Utilizador não tem permissão para carregar capas" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Os identificadores não diferenciam maiúsculas de minúsculas, substituindo o identificador antigo" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadados atualizados com sucesso" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Erro ao editar o livro: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Ficheiro %(file)s enviado" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Formato de origem ou destino para conversão está em falta" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Livro enviado com sucesso para lista de espera de conversão para %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Ocorreu um erro ao converter este livro: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "O livro carregado provavelmente existe na biblioteca, considere alterar antes de carregar novo: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Oops! O Livro selecionado não está disponível. O ficheiro não existe ou não está acessível" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Utilizador não tem permissão para carregar capas" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Os identificadores não diferenciam maiúsculas de minúsculas, substituindo o identificador antigo" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s não é um idioma válido" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadados atualizados com sucesso" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Erro ao editar o livro: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "O livro carregado provavelmente existe na biblioteca, considere alterar antes de carregar novo: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "A extensão de ficheiro '%(ext)s' não pode ser enviada para este servidor" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "A extensão de ficheiro '%(ext)s' não pode ser enviada para este servidor" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "O ficheiro a ser carregado deve ter uma extensão" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "O ficheiro %(filename)s não foi possível ser guardado na pasta temporária" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Falha ao mover ficheiro de capa %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Formato de livro eliminado com sucesso" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Livro eliminado com sucesso" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Não tem permissões para apagar livros" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "editar metadados" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s não é um número válido, ignorando" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Utilizador não tem direitos para carregar formatos de ficheiro adicionais" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Falha ao criar o caminho %(path)s (Permissão negada)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Falha ao armazenar o ficheiro %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Formato de ficheiro %(ext)s adicionado a %(book)s" @@ -650,485 +663,478 @@ msgstr "Configuração do Google Drive não concluída, tente desativar e ativar msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "O domínio Callback não foi verificado. Por favor, siga os passos para verificar o domínio na consola de programadores do Google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "Formato %(format)s não encontrado para o ID do livro: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s não encontrado no Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s não encontrado: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "Enviar para dispositivo de leitura" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Este email foi enviado via Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Email de teste do Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Email de teste" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Comece a usar o Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Email de registo do utilizador: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Converter %(orig)s em %(format)s e enviar para dispositivo de leitura" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Enviar %(format)s para o dispositivo de leitura" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s enviado para o dispositivo de leitura" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Não foi possível ler o ficheiro solicitado. Talvez permissões erradas?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Estatuto de Lido não pode ser alterado: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "A eliminação da pasta de livros do livro %(id)s falhou, o caminho tem subpastas: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Falha ao eliminar livro %(id)s: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Eliminar livro %(id)s apenas da base de dados, caminho do livro inválido: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Renomear autor de: '%(src)s' para '%(dest)s' falhou com o erro: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Ficheiro %(file)s não encontrado no Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Renomear título de: '%(src)s' para '%(dest)s' falhou com o erro: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Caminho do livro %(path)s não encontrado no Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "Encontrada uma conta existente para este endereço de email" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Este nome de utilizador já está registado" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Formato de endereço de email inválido" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "O módulo Python 'advocate' não está instalado, mas é necessário para carregar capas" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Erro ao descarregar a capa" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Erro de formato da capa" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Não possui permissões para aceder a localhost ou à rede local para carregar capas" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Falha em criar um caminho para a capa" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "O ficheiro de capa não é um ficheiro de imagem válido, ou não foi possível ser armazenado" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Apenas ficheiros jpg/jpeg/png/webp/bmp são suportados como ficheiros de capa" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Conteúdo do ficheiro de capa inválido" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Apenas ficheiros jpg/jpeg são suportados como ficheiros de capa" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Capa" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Binário UnRar não encontrado" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Erro a executar UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "DB não é gravável" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Falta de permissões de execução" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Erro a executar UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Enviar todos os livros para lista de espera para cópia de segurança de metadados" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Por favor, aceda ao Calibre-web a partir de um servidor não local para obter uma api_endpoint válida para o dispositivo Kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Configuração Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Autentique-se com %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Sucesso! Agora está autenticado como: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Hiperligação para %(oauth)s bem-sucedida" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Falha na autenticação, nenhum utilizador ligado a uma conta OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Desvincular a %(oauth)s bem-sucedido" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Falha ao desvincular de %(oauth)s" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Não vinculado a %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Falha na autenticação com GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Falha na pesquisa de informações do utilizador no GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Falha na autenticação com Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Falha na pesquisa de informações de utilizador no Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Erro no Oauth do GitHub. Por favor, tente novamente mais tarde." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Erro no Oauth do GitHub: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Erro no Google Oauth, tente novamente mais tarde." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Erro no Oauth do Google: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Estrelas" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Autenticar" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token não encontrado" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "O Token expirou" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Sucesso! Por favor, volte ao seu dispositivo" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Livros" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Mostrar livros recentes" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Livros quentes" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Mostrar livros mais descarregados" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Livros descarregados" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Mostrar livros descarregados" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Top de pontuação de livros" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Mostrar livros mais bem pontuados" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Livros lidos" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Mostrar lido e não lido" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Livros não lidos" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Mostrar Não Lidos" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Descobrir" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Mostrar livros aleatoriamente" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Categorias" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Mostrar secção da categoria" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Séries" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Mostrar secção de séries" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autores" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Mostrar secção de autor" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Editoras" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Mostrar seleção de editoras" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Idiomas" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Mostrar secção de idioma" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Pontuações" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Mostrar secção de pontuações" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formatos de ficheiro" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Mostrar secção de formatos de ficheiro" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Livros arquivados" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Mostrar livros arquivados" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Lista de livros" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Mostrar lista de livros" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Pesquisar" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Publicado depois de " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Publicado antes de " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Pontuação <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Pontuação >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Estado de leitura = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Erro na pesquisa de colunas personalizadas. Por favor, reinicie o Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Pesquisa avançada" @@ -1183,7 +1189,7 @@ msgstr "O livro foi removido da estante: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Desculpe, não possui permissões para remover um livro desta estante" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Criar estante" @@ -1236,45 +1242,45 @@ msgstr "Já existe uma estante pública com o nome '%(title)s' ." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Já existe uma estante privada com o nome'%(title)s' ." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Estante: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Erro ao abrir estante. A estante não existe ou não está acessível" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tarefas" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Aguardando" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Falhado" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Iniciado" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Concluído" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Terminado" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Cancelado" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Estado desconhecido" @@ -1307,178 +1313,178 @@ msgstr "Uma nova atualização está disponível. Clique no botão abaixo para a msgid "No release information available" msgstr "Não existem informações disponíveis sobre o lançamento" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Descobrir (Livros aleatórios)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Livros quentes (Mais descarregados)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Livros descarregados por %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Autor: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Editora: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Séries: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Pontuação: Nenhuma" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Pontuação: %(rating)s estrelas" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Formato do ficheiro: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Categoria: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Idioma: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Descarregamentos" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Lista de pontuações" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Lista de formatos de ficheiro" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Por favor, configure primeiro as configurações de correio SMTP..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Sucesso! Livro enviado para lista de espera para envio a %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Ops! Ocorreu um erro ao enviar este livro: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Ops! Por favor, atualize o seu perfil com um endereço de email válido para envio ao Kindle." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "Por favor, aguarde um minuto para registar o próximo utilizador" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registar" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Ops! O servidor de email não está configurado. Por favor, contacte o seu administrador!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Ops! O servidor de email não está configurado. Por favor, contacte o seu administrador!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Ops! O seu email não é permitido para registo" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Sucesso! O email de confirmação foi enviado para a sua conta de email." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Não é possível ativar a autenticação LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "Por favor, aguarde um minuto antes de nova autenticação" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "agora você está autenticado como: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Autenticação recurso como:'%(nickname)s', servidor LDAP não acessível ou utilizador desconhecido" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Não foi possível autenticar-se: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Nome de utilizador ou senha incorretos" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Nova senha foi enviada para seu endereço de email" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Ocorreu um erro desconhecido. Por favor, tente novamente mais tarde." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Por favor, digite um nome de utilizador válido para poder redefinir a senha" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Agora você está autenticado como: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Perfil de %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Perfil atualizado" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Foi encontrada uma conta já existente com este endereço de email." @@ -1486,54 +1492,58 @@ msgstr "Foi encontrada uma conta já existente com este endereço de email." msgid "Found no valid gmail.json file with OAuth information" msgstr "Não foi encontrado nenhum ficheiro gmail.json válido com informações do OAuth" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s enviado para dispositivo de leitura" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre ebook-convert %(tool)s não encontrado" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "Formato %(format)s não encontrado no disco" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "O conversor de ebook falhou com erro desconhecido" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Conversor Kepubify falhou: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Ficheiro convertido não encontrado ou mais de um ficheiro na pasta %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Conversor de ebook falhou: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre falhou com erro: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Conversor de ebook falhou: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Converter" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "A religar base de dados Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "Email" @@ -1542,30 +1552,26 @@ msgstr "Email" msgid "Backing up Metadata" msgstr "Cópia de segurança de metadados" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "Geradas %(count)s miniaturas para capa" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Miniaturas de capa" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "Geradas {0} miniaturas de série" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "A esvaziar a cache de miniaturas da capa" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Carregar" @@ -1584,11 +1590,11 @@ msgstr "Nome de utilizador" msgid "Email" msgstr "Endereço de email" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "Enviar para o endereço de email do dispositivod e leitura" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Admin" @@ -1598,8 +1604,8 @@ msgstr "Admin" msgid "Password" msgstr "Senha" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Descarregar" @@ -1810,13 +1816,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Cancelar" @@ -1866,16 +1872,76 @@ msgstr "Ordenar de acordo com a data de publicação, o mais novo primeiro" msgid "Sort according to publishing date, oldest first" msgstr "Ordenar de acordo com a data de publicação, o mais antigo primeiro" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "reduzir" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Outros por" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Livro %(index)s de %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Idioma" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Editora" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publicado em" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Descrição:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Anterior" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Seguinte" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Nenhum resultado encontrado" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Entrada" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Pesquisar na biblioteca" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Sair" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Apagar livro" @@ -1904,99 +1970,107 @@ msgstr "Converter para:" msgid "Convert book" msgstr "Converter livro" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "A carregar..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Fechar" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Erro" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Carregamento concluído, a processar. Por favor, aguarde ..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Formato de carregamento" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Título do livro" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Etiquetas" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Identificador da série" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Data de publicação" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Pontuação" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Descrição" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identificadores" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Tipo de identificador" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Valor do identificador" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Remover" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Adicionar identificador" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Etiquetas" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Identificador da série" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Pontuação" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Obter capa a partir de URL (JPEG - Imagem será descarregada e armazenada na base de dados)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Carregar capa a partir de disco local" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Data de publicação" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Editora" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Idioma" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Sim" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Não" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Formato de carregamento" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Ver livro ao guardar" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Obter metadados" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2004,37 +2078,31 @@ msgstr "Obter metadados" msgid "Save" msgstr "Guardar" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Termo-chave" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Pesquisar termo-chave" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Clique na capa para carregar os metadados para o formulário" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "A carregar..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Fechar" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Fonte" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Erro de pesquisa!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nenhum resultado encontrado! Por favor, tente pesquisar por outro termo-chave." @@ -2141,7 +2209,7 @@ msgid "Enter " msgstr "Preencher " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Tem realmente a certeza?" @@ -2617,74 +2685,61 @@ msgstr "Adicionar etiquetas Permitidas/Negadas" msgid "Add Allowed/Denied custom column values" msgstr "Adicionar valores permitidos/negados de coluna personalizada" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Ler no Navegador" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Ouvir no Navegador" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Livro %(index)s de %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publicado em" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Marcar como não Lido" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Marcar como lido" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Marcar como não Lido" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Lido" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Restaurar do ficheiro" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Adicionar ao ficheiro" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Arquivado" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Descrição:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Adicionar à estante" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Público)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Editar metadados" @@ -2757,10 +2812,6 @@ msgstr "Digite o nome do domínio" msgid "Denied Domains (Blacklist)" msgstr "Domínios ngados (Lista Negra)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Seguinte" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Abra o ficheiro .kobo/Kobo/Kobo eReader.conf em um editor de texto e adicione (ou edite):" @@ -2781,11 +2832,16 @@ msgstr "Instancia do Calibre-Web não configurada. Por favor, contacte o seu adm msgid "Create Issue" msgstr "Criar ocorrência" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Configuração da base de dados" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Voltar para entrada" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Terminar sessão" @@ -2875,7 +2931,7 @@ msgstr "Livros ordenados por pontuação" msgid "Books ordered by file formats" msgstr "Livros ordenados por formatos de ficheiro" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Estantes" @@ -2884,60 +2940,37 @@ msgstr "Estantes" msgid "Books organized in shelves" msgstr "Livros organizados em estantes" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Entrada" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Alternar navegação" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Pesquisar na biblioteca" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Simples" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Conta" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Sair" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "A carregar..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Erro" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Carregamento concluído, a processar. Por favor, aguarde ..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Configurações" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Por favor, não refresque a página" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Navegar" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Sobre" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Anterior" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Detalhes do livro" @@ -3053,7 +3086,7 @@ msgstr "Pasta superior" msgid "Select" msgstr "Selecionar" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3061,34 +3094,80 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Catálogo de ebooks Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "leitor de epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Escolher um nome de utilizador" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Claro" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Escuro" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sépia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Preto" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Refluir o texto quando as barras laterais estiverem abertas." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "Tamanhos de letra" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Apagar" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Aguardando" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Vertical" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Lido" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Coluna Lido é inválida" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Leitor de banda desenhada" @@ -3266,10 +3345,6 @@ msgstr "Esta ligação de verificação irá expirar em 10 minutos." msgid "Generate Series Cover Thumbnails" msgstr "Gerar miniaturas para capa de série" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Nenhum resultado encontrado" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Termo de pesquisa:" @@ -3286,11 +3361,11 @@ msgstr "Data de publicação de" msgid "Published Date To" msgstr "Data de publicação até" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3326,11 +3401,13 @@ msgstr "Pontuação acima" msgid "Rating Below" msgstr "Pontuação abaixo" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "De:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Para:" @@ -3354,6 +3431,16 @@ msgstr "Desativar alterar ordenação" msgid "Enable Change order" msgstr "Ativar alterar ordenação" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Ordenar de acordo com a data do livro, o mais recente primeiro" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Ordenar de acordo com a data do livro, o mais antigo primeiro" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Partilhar com todos" @@ -3422,15 +3509,20 @@ msgstr "Progresso" msgid "Run Time" msgstr "Tempo de execução" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Fundir" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Ações" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Esta tarefa será cancelada. Qualquer progresso feito por esta tarefa será guardado." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Se esta for uma tarefa agendada, ela será executada novamente durante o próximo horário agendado." @@ -3438,6 +3530,10 @@ msgstr "Se esta for uma tarefa agendada, ela será executada novamente durante o msgid "Reset user Password" msgstr "Redefinir senha do utilizador" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Idioma dos livros" diff --git a/cps/translations/pt_BR/LC_MESSAGES/messages.mo b/cps/translations/pt_BR/LC_MESSAGES/messages.mo index d7615a4d..6fa1d40a 100644 Binary files a/cps/translations/pt_BR/LC_MESSAGES/messages.mo and b/cps/translations/pt_BR/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/pt_BR/LC_MESSAGES/messages.po b/cps/translations/pt_BR/LC_MESSAGES/messages.po index fb830ba1..135e6fc0 100644 --- a/cps/translations/pt_BR/LC_MESSAGES/messages.po +++ b/cps/translations/pt_BR/LC_MESSAGES/messages.po @@ -4,7 +4,7 @@ msgid "" msgstr "" "Project-Id-Version: PROJECT VERSION\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" "Last-Translator: FULL NAME \n" "Language: br\n" @@ -13,501 +13,519 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Estatísticas" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Servidor reiniciado, por favor recarregue a página" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Executando o desligamento do servidor, por favor feche a janela" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Comando desconhecido" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "E-mail de teste enfileirado para envio para %(email)s, verifique o resultado em Tarefas" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Desconhecido" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Página de Administração" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Configuração Básica" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Configuração de UI" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "A Coluna Personalizada No.%(column)d não existe no banco de dados do calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Editar Usuários" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Todos" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Usuário não encontrado" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} usuário(s) deletedos com sucesso" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Mostrar Tudo" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Requisição Malformada" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Nome do Convidado não pode ser alterado" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Convidado não pode ter esta função" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Nenhum usuário administrador restante, impossível remover a função de administrador" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Valor tem de ser Verdadeiro ou Falso" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Função Inválida" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Convidado não pode ter esta visão" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Visão Inválida" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "O idioma do Convidado é detectado automaticamente e não pode ser alterado" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Nenhum Idioma Válido Fornecido" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Nenhum Idioma do Livro Válido Fornecido" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parametro não encontrado" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Coluna Lido Inválida" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Coluna Restrito Inválida" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Configuração do Calibre-Web atualizada" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Você realmente quer apagar a Kobo Token?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Você realmente quer apagar este domínio?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Você realmente quer apagar este usuário?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Tem certeza que quer apagar essa estante?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Tem certeza que quer alterar o idioma do(s) usuário(s) selecionados?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Tem certeza que quer alterar os idiomas de livros visíveis par o usuário(s) selecionado(s)?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Tem certeza que quer alterar a função selecionada para o(s) usuário(s) selecionado(s)?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Tem certeza que quer alterar as restriçõeo selecionada para o(s) usuário(s) selecionado(s)?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Tem certeza de que quer alterar as restrições de visibilidade selecionadas para os usuários selecionados?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Tem certeza de que quer alterar o comportamento de sincronização da estante para o usuário selecionado?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Tem certeza que queres alterar a localização da biblioteca Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "O Calibre-Web buscará por Capas atualizadas e atualizará as Miniaturas de Capas, isso pode demorar um pouco" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Tem certeza de que deseja apagar o banco de dados de sincronização do Calibre-Web para forçar uma sincronização completa com seu Kobo Reader?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Negar" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Permitir" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} entradas de sincronização deletadas" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Tag não encontrada" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Ação Inválida" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json Não Está Configurado para Aplicação Web" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "A localização do arquivo de log não é válida, digite o caminho correto" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "A localização do arquivo de log de acesso não é válida, digite o caminho correto" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Digite um provedor LDAP, porta, DN e identificador de objeto do usuário" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Por favor, digite uma Conta de Serviço LDAP e Senha" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Por favor, digite uma Conta de Serviço LDAP" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "O filtro de objeto de grupo LDAP precisa ter um identificador de formato \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Filtro de objeto de grupo LDAP tem parênteses incomparáveis" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "O filtro de objeto de usuário LDAP precisa ter um identificador de formato \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Filtro de objeto de usuário LDAP tem parênteses incomparáveis" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "O filtro de usuário membro do LDAP precisa ter um identificador de formato \"%s\"" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Filtro de usuário de membro LDAP tem parênteses incomparáveis" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "Localização de LDAP CACertificate, Certificados ou Key Inválida, Insira o Caminho Correto" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Adicionar Novo Usuário" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Editar configurações do servidor de e-mail" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Erro de banco de dados: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "E-mail de teste enfileirado para envio para %(email)s, verifique o resultado em Tarefas" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Ocorreu um erro ao enviar o e-mail de teste: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Por favor, configure seu endereço de e-mail primeiro..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Atualização das configurações do servidor de e-mail" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Editar configurações de tarefas agendadas" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Hora de início inválida para a tarefa especificada" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Duração inválida para a tarefa especificada" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Configurações de tarefas agendadas atualizadas" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Ocorreu um erro desconhecido. Por favor, tente novamente mais tarde." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Settings DB não é gravável" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Editar Usuário %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Senha do usuário %(user)s redefinida" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Por favor, configure primeiro as configurações de correio SMTP..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Visualizador do Log" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Solicitação de pacote de atualização" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Baixando pacote de atualização" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Descompactando pacote de atualização" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Substituindo arquivos" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "As conexões à base de dados estão fechadas" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Parando servidor" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Atualização concluída, pressione okay e recarregue a página" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Atualização falhou:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Erro HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Erro de conexão" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Tempo limite durante o estabelecimento da conexão" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Erro geral" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "Arquivo de atualização não pôde ser salvo no diretório temporário" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Arquivos não puderam ser substituídos durante a atualização" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Falha ao extrair pelo menos um usuário LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Falha ao criar pelo menos um usuário LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Erro: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Erro: Nenhum usuário retornado na resposta do servidor LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "No mínimo um usuário LDAP não encontrado no banco de dados" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} Usuário Importado com Sucesso" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "A localização do banco de dados não é válida, digite o caminho correto" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "DB não é gravável" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Localização do Keyfile Inválida, Insira o Caminho Correto" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Localização do Certfile Inválida, Insira o Caminho Correto" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Configurações do Banco de Dados Atualizada" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Configuração do Banco de Dados" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Por favor, preencha todos os campos!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "O e-mail não é de um domínio válido" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Adicionar novo usuário" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Usuário '%(user)s' criado" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Encontrada uma conta existente para este endereço de e-mail ou apelido." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Usuário '%(nick)s' excluído" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Impossível excluir Convidado" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Nenhum usuário administrador restante, não é possível apagar o usuário" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Usuário '%(nick)s' atualizado" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Pesquisar" + #: cps/converter.py:31 msgid "not installed" msgstr "não instalado" @@ -516,128 +534,123 @@ msgstr "não instalado" msgid "Execution permissions missing" msgstr "Faltam as permissões de execução" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "A Coluna Personalizada No.%(column)d não existe no banco de dados do calibre" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Nenhum" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Oops! O Livro selecionado não está disponível. O arquivo não existe ou não é acessível" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Usuário não tem permissão para fazer upload da capa" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Os identificadores não diferenciam maiúsculas de minúsculas, substituindo o identificador antigo" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadados atualizados com sucesso" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Erro ao editar o livro: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Arquivo %(file)s enviado" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Formato de origem ou destino para conversão ausente" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Livro enfileirado com sucesso para conversão em %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Ocorreu um erro ao converter este livro: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "O livro carregado provavelmente existe na biblioteca, considere alterar antes de carregar novo: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Oops! O Livro selecionado não está disponível. O arquivo não existe ou não é acessível" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Usuário não tem permissão para fazer upload da capa" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Os identificadores não diferenciam maiúsculas de minúsculas, substituindo o identificador antigo" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s não é um idioma válido" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadados atualizados com sucesso" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Erro ao editar o livro: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "O livro carregado provavelmente existe na biblioteca, considere alterar antes de carregar novo: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "A extensão de arquivo '%(ext)s' não pode ser enviada para este servidor" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "A extensão de arquivo '%(ext)s' não pode ser enviada para este servidor" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "O arquivo a ser carregado deve ter uma extensão" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "O arquivo %(filename)s não pôde ser salvo no diretório temporário" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Falha ao mover arquivo de capa %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Formato do Livro Apagado com Sucesso" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Livro Apagado com Sucesso" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Você não tem permissão para apagar livros" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "editar metadados" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s não é um número válido, ignorando" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Usuário não tem direitos para fazer upload de formatos de arquivo adicionais" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Falha ao criar o caminho %(path)s (Permission denied)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Falha ao armazenar o arquivo %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Formato de arquivo %(ext)s adicionado a %(book)s" @@ -650,485 +663,478 @@ msgstr "Configuração do Google Drive não concluída, tente desativar e ativar msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "O domínio Callback não foi verificado, por favor siga os passos para verificar o domínio no console do desenvolvedor do google" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "Formato %(format)s não encontrado para o id do livro: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s não encontrado no Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s não encontrado: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "Enviar para E-Reader" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Este e-mail foi enviado via Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "E-mail de teste do Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "E-mail de teste" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Comece a usar o calibre-web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "E-mail de registro do usuário: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Converter %(orig)s em %(format)s e enviar para E-Reader" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Enviar %(format)s para o E-Reader" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s enviado para E-Reader" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "O arquivo solicitado não pôde ser lido. Talvez permissões erradas?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Status Lido não pode ser alterado: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "A exclusão da pasta de livros do livro %(id)s falhou, o caminho tem subpastas: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Falha ao excluir livro %(id)s: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Excluindo livro %(id)s somente do banco de dados, caminho do livro inválido: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Renomear autor de: '%(src)s' para '%(dest)s' falhou com o erro: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Arquivo %(file)s não encontrado no Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Renomear título de: '%(src)s' para '%(dest)s' falhou com o erro: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Caminho do livro %(path)s não encontrado no Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Este nome de usuário já está registrado" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Formato de endereço de e-mail inválido" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "O módulo Python 'advocate' não está instalado, mas é necessário para uploads de capa" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Erro ao Baixar a capa" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Erro de Formato da Capa" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Você não tem permissão para acessar localhost ou a rede local para uploads de capa" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Falha em criar caminho para a capa" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "O arquivo de capa não é um arquivo de imagem válido, ou não pôde ser armazenado" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Apenas arquivos jpg/jpeg/png/webp/bmp são suportados como arquivos de capa" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Conteúdo do arquivo de capa inválido" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Apenas arquivos jpg/jpeg são suportados como arquivos de capa" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Capa" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Binário UnRar não encontrado" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Erro excecutando UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "DB não é gravável" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Faltam as permissões de execução" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Erro excecutando UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Por favor, acesse o calibre-web de um host não local para obter um api_endpoint válido para o dispositivo kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Configuração Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Registre-se com %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "agora você está logado como: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Link para %(oauth)s bem-sucedido" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Login falhou, nenhum usuário ligado a uma conta OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Desvincular para %(oauth)s bem-sucedido" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Falha ao desvincular para %(oauth)s" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Não vinculado a %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Falha no login com o GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Falha na busca de informações do usuário no GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Falha no login com o Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Falha na busca de informações de usuário no Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Erro no Oauth do GitHub, tente novamente mais tarde." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Erro no Oauth do GitHub: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Erro no Google Oauth, tente novamente mais tarde." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Erro no Oauth do Google: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Estrelas" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Login" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token não encontrado" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "O Token expirou" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Sucesso! Por favor, volte ao seu aparelho" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Livros" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Mostrar livros recentes" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Livros Quentes" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Mostrar Livros Quentes" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Livros Baixados" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Mostrar Livros Baixados" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Livros Mais Bem Avaliados" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Mostrar Livros Mais Bem Avaliados" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Livros Lidos" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Mostrar lido e não lido" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Livros Não Lidos" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Mostrar Não Lidos" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Descubra" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Mostrar Livros Aleatórios" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Categorias" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Mostrar seção de categoria" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Série" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Mostrar seção de séries" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autores" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Mostrar seção de autor" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Editoras" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Mostrar seção de editoras" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Idiomas" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Mostrar seção de idioma" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Avaliações" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Mostrar seção de avaliações" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Formatos de arquivo" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Mostrar seção de formatos de arquivo" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Livros Arquivados" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Mostrar livros arquivados" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Lista de Livros" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Mostrar Lista de Livros" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Pesquisar" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Publicado depois de " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Publicado antes de " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Avaliação <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Avaliação >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Status de leitura = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Erro na pesquisa de colunas personalizadas, reinicie o Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Pesquisa Avançada" @@ -1183,7 +1189,7 @@ msgstr "O livro foi removido da estante: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Desculpe, você não tem permissão para remover um livro desta estante" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Crie uma Estante" @@ -1236,45 +1242,45 @@ msgstr "Já existe uma estante pública com o nome '%(title)s' ." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Já existe uma estante privada com o nome'%(title)s' ." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Estante: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Erro ao abrir estante. A estante não existe ou não está acessível" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Tarefas" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Aguardando" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Falha" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Iniciado" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Concluído" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Terminado" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Cancelado" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Status Desconhecido" @@ -1307,178 +1313,178 @@ msgstr "Uma nova atualização está disponível. Clique no botão abaixo para a msgid "No release information available" msgstr "Não há informações de lançamento disponíveis" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Descobrir (Livros Aleatórios)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Livros Quentess (Mais Baixados)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Livros baixados por %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Autor: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Editora: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Série: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Avaliação: Nenhuma" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Avaliação: %(rating)s estrelas" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Formato do arquivo: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Categoria: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Idioma: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Downloads" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Lista de Avaliações" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Lista de formatos de arquivo" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Por favor, configure primeiro as configurações de correio SMTP..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Livro enfileirado com sucesso para envio para %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Ops! Ocorreu um erro ao enviar este livro: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Por favor, atualize seu perfil com um endereço de e-mail Envie Para o Kindle válido." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registe-se" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "O servidor de E-Mail não está configurado, por favor contacte o seu administrador!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "O servidor de E-Mail não está configurado, por favor contacte o seu administrador!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Seu e-mail não tem permissão para registrar" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "O e-mail de confirmação foi enviado para a sua conta de e-mail." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Não é possível ativar a autenticação LDAP" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "agora você está logado como: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Login de reserva como:'%(nickname)s', servidor LDAP não acessível ou usuário desconhecido" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Não foi possível fazer o login: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Nome de Usuário ou Senha incorretos" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Nova Senha foi enviada para seu endereço de e-mail" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Ocorreu um erro desconhecido. Por favor, tente novamente mais tarde." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Por favor, digite um nome de usuário válido para redefinir a senha" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "agora você está logado como: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Perfil de %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Perfil atualizado" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Encontrada uma conta existente para este endereço de e-mail." @@ -1486,54 +1492,58 @@ msgstr "Encontrada uma conta existente para este endereço de e-mail." msgid "Found no valid gmail.json file with OAuth information" msgstr "Não foi encontrado nenhum arquivo gmail.json válido com informações do OAuth" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s enviado para E-Reader" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre ebook-convert %(tool)s não encontrado" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "Formato %(format)s não encontrado no disco" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "O conversor de Ebook falhou com erro desconhecido" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-converter falhou: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Arquivo convertido não encontrado ou mais de um arquivo na pasta %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ebook-converter falhou: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre falhou com erro: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ebook-converter falhou: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Converter" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Reconectando banco de dados Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "E-mail" @@ -1542,30 +1552,26 @@ msgstr "E-mail" msgid "Backing up Metadata" msgstr "Cópia de segurança de metadados" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "Gerado %(count)s miniaturas de capa" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Miniatura da Capa" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "Gerado {0} miniaturas de série" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Limpando o cache de miniaturas da capa" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Upload" @@ -1584,11 +1590,11 @@ msgstr "Nome de usuário" msgid "Email" msgstr "Endereço de e-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "Enviar para o endereço de e-mail do E-Reader" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Admin" @@ -1598,8 +1604,8 @@ msgstr "Admin" msgid "Password" msgstr "Senha" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Baixar" @@ -1810,13 +1816,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Cancelar" @@ -1866,16 +1872,76 @@ msgstr "Classificar de acordo com a data de publicação, o mais novo primeiro" msgid "Sort according to publishing date, oldest first" msgstr "Classificar de acordo com a data de publicação, primeiro mais antigo" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "reduzir" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Mais por" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Livro %(index)s de %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Idioma" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Editora" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publicado em" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Descrição:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Anterior" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Próximo" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Nenhum resultado encontrado" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Início" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Pesquisar na Biblioteca" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Sair" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Apagar Livro" @@ -1904,99 +1970,107 @@ msgstr "Converter para:" msgid "Convert book" msgstr "Converter livro" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Enviando..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Fechar" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Erro" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Upload concluído, processando, por favor aguarde ..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Formato de upload" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Título do Livro" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Tags" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Identificação da série" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Data de Publicação" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Avaliação" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Descrição" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identificadores" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Tipo de identificador" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Valor do Identificador" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Remover" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Adicionar Identificador" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Tags" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Identificação da série" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Avaliação" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Buscar capa na URL (JPEG - Imagem será baixada e armazenada na base de dados)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Upload de Capa do Disco Local" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Data de Publicação" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Editora" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Idioma" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Sim" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Não" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Formato de upload" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Ver Livro ao Salvar" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Buscar Metadados" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2004,37 +2078,31 @@ msgstr "Buscar Metadados" msgid "Save" msgstr "Salvar" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Palavra-chave" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Pesquisar palavra-chave" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Clique na capa para carregar os metadados para o formulário" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Carregando..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Fechar" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Fonte" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Erro de busca!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nenhum resultado(s) encontrado(s)! Por favor, tente outra palavra-chave." @@ -2141,7 +2209,7 @@ msgid "Enter " msgstr "Entrar " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Você realmente tem certeza?" @@ -2617,74 +2685,61 @@ msgstr "Adicionar Tags Permitidas/Negadas" msgid "Add Allowed/Denied custom column values" msgstr "Adicionar Valores Permitidos/Negados de Coluna Personalizada" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Ler no Browser" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Ouvir no Browser" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Livro %(index)s de %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publicado em" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Marcar como Não Lido" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Marcar como Lido" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Marcar como Não Lido" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Lido" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Restaurar do arquivo" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Adicionar ao arquivo" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Arquivado" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Descrição:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Adicionar à estante" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Público)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Editar Metadados" @@ -2757,10 +2812,6 @@ msgstr "Digite o nome do domínio" msgid "Denied Domains (Blacklist)" msgstr "Domínios Negados (Lista Negra)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Próximo" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Abra o arquivo .kobo/Kobo/Kobo eReader.conf em um editor de texto e adicione (ou edite):" @@ -2781,11 +2832,16 @@ msgstr "Instancia do Calibre-Web não configurado, por favor contacte o seu admi msgid "Create Issue" msgstr "Criar Issue" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Configuração do Banco de Dados" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Voltar para Início" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Deslogar" @@ -2875,7 +2931,7 @@ msgstr "Livros ordenados por Avaliação" msgid "Books ordered by file formats" msgstr "Livros ordenados por formatos de arquivo" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Estantes" @@ -2884,60 +2940,37 @@ msgstr "Estantes" msgid "Books organized in shelves" msgstr "Livros organizados em Estantes" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Início" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Alternar Navegação" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Pesquisar na Biblioteca" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Simples" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Conta" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Sair" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Enviando..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Erro" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Upload concluído, processando, por favor aguarde ..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Configurações" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Por favor, não atualize a página" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Navegue em" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Sobre" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Anterior" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Detalhes do Livro" @@ -3053,7 +3086,7 @@ msgstr "Diretório Acima" msgid "Select" msgstr "Selecione" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3061,34 +3094,80 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Catálogo de e-books Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "leitor de epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Escolha um nome de usuário" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Claro" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Escuro" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sépia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Preto" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Refluir o texto quando as barras laterais estiverem abertas." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Apagar" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Aguardando" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Vertical" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Lido" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Coluna Lido Inválida" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Leitor de Quadrinhos" @@ -3266,10 +3345,6 @@ msgstr "Este link de verificação irá expirar em 10 minutos." msgid "Generate Series Cover Thumbnails" msgstr "Gerar miniaturas de capa de Série" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Nenhum resultado encontrado" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Termo de busca:" @@ -3286,11 +3361,11 @@ msgstr "Data de Publicação de" msgid "Published Date To" msgstr "Data de Publicação Até" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3326,11 +3401,13 @@ msgstr "Avaliação Acima" msgid "Rating Below" msgstr "Avaliação Abaixo" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "De:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Para:" @@ -3354,6 +3431,16 @@ msgstr "Desabilitar Alterar Ordem" msgid "Enable Change order" msgstr "Habilitar Alterar Ordem" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Classificar de acordo com a data do livro, o mais recente primeiro" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Classificar de acordo com a data do livro, o mais antigo primeiro" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Compartilhar com Todos" @@ -3422,15 +3509,20 @@ msgstr "Progresso" msgid "Run Time" msgstr "Tempo de execução" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Mesclar" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Ações" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Esta tarefa será cancelada. Qualquer progresso feito por esta tarefa será salvo." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Se esta for uma tarefa agendada, ela será executada novamente durante o próximo horário agendado." @@ -3438,6 +3530,10 @@ msgstr "Se esta for uma tarefa agendada, ela será executada novamente durante o msgid "Reset user Password" msgstr "Redefinir senha do usuário" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Idioma dos Livros" diff --git a/cps/translations/ru/LC_MESSAGES/messages.mo b/cps/translations/ru/LC_MESSAGES/messages.mo index 0f716462..f2d057e8 100644 Binary files a/cps/translations/ru/LC_MESSAGES/messages.mo and b/cps/translations/ru/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/ru/LC_MESSAGES/messages.po b/cps/translations/ru/LC_MESSAGES/messages.po index cb38d481..f5061e7e 100644 --- a/cps/translations/ru/LC_MESSAGES/messages.po +++ b/cps/translations/ru/LC_MESSAGES/messages.po @@ -8,7 +8,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2020-04-29 01:20+0400\n" "Last-Translator: ZIZA\n" "Language: ru\n" @@ -17,512 +17,530 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Статистика" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Сервер перезагружен, пожалуйста, обновите страницу" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Производится остановка сервера, пожалуйста, закройте окно" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Неизвестная команда" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Книга успешно поставлена в очередь для отправки на %(eReadermail)s" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Неизвестно" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Администрирование" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Настройки сервера" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Настройка интерфейса" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Управление сервером" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Все" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Показать все" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Конфигурация Calibre-Web обновлена" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Вы действительно хотите удалить Kobo Token ?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Вы действительно хотите удалить эту книжную полку?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Вы действительно хотите удалить эту книжную полку?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Вы действительно хотите удалить эту книжную полку?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Вы действительно хотите удалить эту книжную полку?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Вы действительно хотите остановить Calibre-Web?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Запретить" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Разрешить" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json не настроен для веб-приложения" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Неправильное расположение файла журнала, пожалуйста, введите правильный путь." -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Недопустимое расположение файла журнала доступа, пожалуйста, введите правильный путь" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Пожалуйста, введите провайдера LDAP, порт, DN и идентификатор объекта пользователя" -#: cps/admin.py:1211 +#: cps/admin.py:1223 #, fuzzy msgid "Please Enter a LDAP Service Account and Password" msgstr "Пожалуйста, введите действительное имя пользователя для сброса пароля" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Фильтр объектов группы LDAP должен иметь один идентификатор формата \"%s\"" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Фильтр объектов группы LDAP имеет незавершённые круглые скобки" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Фильтр объектов пользователя LDAP должен иметь один идентификатор формата \"%s\"" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Фильтр объектов пользователя LDAP имеет незавершенную круглую скобку" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Добавить нового пользователя" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Изменить настройки SMTP" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Произошла ошибка при отправке тестового письма на: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Пожалуйста, сначала настройте свой адрес электронной почты ..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Настройки E-mail сервера обновлены" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Неизвестная ошибка. Попробуйте позже." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Изменить пользователя %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Пароль для пользователя %(user)s сброшен" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Пожалуйста, сперва настройте параметры SMTP....." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Просмотр лога" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Проверка обновлений" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Загрузка обновлений" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Распаковка обновлений" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Замена файлов" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Соединения с базой данных закрыты" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Остановка сервера" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Обновления установлены, нажмите ок и перезагрузите страницу" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Ошибка обновления:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Ошибка HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Ошибка соединения" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Тайм-аут при установлении соединения" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Общая ошибка" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Не удалось сохранить файл обновления во временной папке." -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Не удалось создать хотя бы одного пользователя LDAP" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Не удалось создать хотя бы одного пользователя LDAP" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Ошибка: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Ошибка: ни одного пользователя не найдено в ответ на запрос сервер LDAP" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "По крайней мере, один пользователь LDAP не найден в базе данных" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Расположение Базы Данных неверно, пожалуйста, введите правильный путь." -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Расположение ключевого файла неверно, пожалуйста, введите правильный путь" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Расположение Certfile не является действительным, пожалуйста, введите правильный путь" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "Настройки E-mail сервера обновлены" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Дополнительный Настройки" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Пожалуйста, заполните все поля!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-mail не из существующей доменной зоны" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Добавить пользователя" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Пользователь '%(user)s' добавлен" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Для этого адреса электронной почты или логина уже есть учётная запись." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Пользователь '%(nick)s' удалён" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Это последний администратор, невозможно удалить пользователя" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Пользователь '%(nick)s' обновлён" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Поиск" + #: cps/converter.py:31 msgid "not installed" msgstr "не установлено" @@ -531,128 +549,123 @@ msgstr "не установлено" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Нет" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Невозможно открыть книгу. Файл не существует или недоступен" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Метаданные обновлены" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Файл %(file)s загружен" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Исходный или целевой формат для конвертирования отсутствует" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Книга успешно поставлена в очередь для конвертирования в %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Произошла ошибка при конвертирования этой книги: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Загруженная книга, вероятно, существует в библиотеке, перед тем как загрузить новую, рассмотрите возможность изменения: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Невозможно открыть книгу. Файл не существует или недоступен" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s не допустимый язык" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Метаданные обновлены" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Загруженная книга, вероятно, существует в библиотеке, перед тем как загрузить новую, рассмотрите возможность изменения: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Запрещена загрузка файлов с расширением '%(ext)s'" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Запрещена загрузка файлов с расширением '%(ext)s'" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Загружаемый файл должен иметь расширение" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Файл %(filename)s не удалось сохранить во временную папку" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "изменить метаданные" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Ошибка при создании пути %(path)s (Доступ запрещён)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Не удалось сохранить файл %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Формат файла %(ext)s добавлен в %(book)s" @@ -665,483 +678,476 @@ msgstr "Настройка Google Drive не завершена, попробу msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Не удалось проверить домен обратного вызова, пожалуйста, выполните шаги для проверки домена в консоли разработчика Google." -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s форма не найден для книги с id: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s не найден на Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s не найден: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Отправить на Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Это электронное письмо было отправлено через Caliber-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Тестовый e-mail для Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Тестовый e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Начать работу с Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Регистрационный e-mail для пользователя: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Преобразовать %(orig)s в %(format)s и отправить в Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Отправить %(format)s в Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Отправить на Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Запрашиваемый файл не может быть прочитан. Возможно у вас нет разрешения?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Переименовывание заголовка с: '%(src)s' на '%(dest)s' не удалось из-за ошибки: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Файл %(file)s не найден на Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Переименовывание заголовка с: '%(src)s' на '%(dest)s' не удалось из-за ошибки: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Путь книги %(path)s не найден на Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Это имя пользователя уже занято" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Не удалось создать путь для обложки." -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Только файлы в формате jpg / jpeg поддерживаются как файл обложки" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Обзор" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Пожалуйста, подключитесь к Calibre-Web не с локального хоста, чтобы получить действительный api_endpoint для устройства Kobo" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Настройка Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Зарегистрируйтесь с %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "вы вошли как пользователь '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Связь с %(oauth)s Успешна" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Не удалось войти, ни один пользователь не связан с учетной записью OAuth" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Отмена связи с %(oauth)s успешно" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Связь с %(oauth)s не удалось отмененить" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Не удалось войти в систему с помощью GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Не удалось получить информацию о пользователе из GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Не удалось войти в систему с помощью Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Не удалось получить информацию о пользователе из Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Ошибка GitHub Oauth, пожалуйста попробуйте позже." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Ошибка Google Oauth, пожалуйста попробуйте позже." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Логин" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Ключ не найден" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Ключ просрочен" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Успешно! Пожалуйста, проверьте свое устройство" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Книги" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Показывать недавние книги" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Популярные Книги" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Показывать популярные книги" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Книги с наилучшим рейтингом" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Показывать книги с наивысшим рейтингом" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Прочитанные Книги" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Показывать прочитанные и непрочитанные" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Непрочитанные Книги" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Показать непрочитанное" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Обзор" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Показывать Случайные Книги" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Категории" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Показывать выбор категории" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Серии" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Показывать выбор серии" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Авторы" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Показывать выбор автора" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Издатели" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Показать выбор издателя" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Языки" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Показывать выбор языка" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Рейтинги" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Показать выбор рейтинга" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Форматы файлов" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Показать выбор форматов файлов" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Показывать недавние книги" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Поиск" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Опубликовано после " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Опубликовано до " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Рейтинг <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Рейтинг >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Расширенный поиск" @@ -1197,7 +1203,7 @@ msgstr "Книга удалена с полки: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Создать полку" @@ -1252,45 +1258,45 @@ msgstr "Публичная полка с названием '%(title)s' уже msgid "A private shelf with the name '%(title)s' already exists." msgstr "Приватная полка с названием '%(title)s' уже существует." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Полка: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Ошибка открытия Полки. Полка не существует или недоступна" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Задания" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Ожидание" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Неудачно" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Начало" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Завершено" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Неизвестный статус" @@ -1323,178 +1329,178 @@ msgstr "Новое обновление доступно. Нажмите на к msgid "No release information available" msgstr "Информация о выпуске недоступна" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Обзор (Случайные Книги)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Популярные книги (часто загружаемые)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Автор: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Издатель: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Серии: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Оценка: %(rating)s звезды(а)" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Формат файла: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Категория: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Язык: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Скачать" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Список рейтингов" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Список форматов файлов" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Пожалуйста, сперва настройте параметры SMTP....." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Книга успешно поставлена в очередь для отправки на %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "При отправке этой книги произошла ошибка: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Пожалуйста, сначала настройте e-mail на вашем kindle..." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Зарегистрироваться" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Сервер электронной почты не настроен, обратитесь к администратору !" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Сервер электронной почты не настроен, обратитесь к администратору !" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Ваш e-mail не подходит для регистрации" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Письмо с подтверждением отправлено вам на e-mail." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Не удается активировать LDAP аутентификацию" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "вы вошли как пользователь '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Резервный вход в систему как: '%(nickname)s', LDAP-сервер недоступен или пользователь не известен" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Не удалось войти: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Ошибка в имени пользователя или пароле" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Новый пароль был отправлен на ваш адрес электронной почты" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Неизвестная ошибка. Попробуйте позже." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Пожалуйста, введите действительное имя пользователя для сброса пароля" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "вы вошли как пользователь '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Профиль %(name)s's" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Профиль обновлён" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Этот адрес электронной почты уже зарегистрирован." @@ -1503,54 +1509,58 @@ msgstr "Этот адрес электронной почты уже зарег msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Отправить на Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Ошибка Ebook-конвертора: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ошибка Ebook-конвертора: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1559,30 +1569,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "изменить метаданные" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Загрузить" @@ -1601,12 +1607,12 @@ msgstr "Имя пользователя" msgid "Email" msgstr "Адрес электронной почты" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Отправить на Kindle Адрес электронной почты" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Управление" @@ -1616,8 +1622,8 @@ msgstr "Управление" msgid "Password" msgstr "Пароль" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Скачать" @@ -1828,13 +1834,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Отмена" @@ -1884,16 +1890,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "уменьшить" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Ещё от" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Язык" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Издатель" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Опубликованный" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Описание:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Предыдущий" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Далее" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Результаты не найдены" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Главная" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Поиск в библиотеке" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Выход" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Удалить книгу" @@ -1922,99 +1988,107 @@ msgstr "Конвертировать в:" msgid "Convert book" msgstr "Конвертировать книгу" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Загружается..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Закрыть" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Ошибка" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Загрузка завершена, обработка, пожалуйста, подождите..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Загружаемый формат" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Название книги" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Автор" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Теги" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID Серии" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Опубликовано" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Рейтинг" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Описание" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Теги" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID Серии" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Рейтинг" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "URL обложки(jpg, обложка загружается и сохраняется в базе данных, после этого поле снова пустое)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Загрузить обложку с диска" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Опубликовано" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Издатель" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Язык" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Да" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Нет" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Загружаемый формат" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Просмотреть книгу после сохранения" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Получить метаданные" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2022,38 +2096,32 @@ msgstr "Получить метаданные" msgid "Save" msgstr "Сохранить" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Ключевое слово" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Поиск по ключевому слову " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Нажмите на обложку, чтобы получить метаданные" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Загрузка..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Закрыть" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Источник" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Ошибка поиска!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Результат(ы) не найдены! Попробуйте другое ключевое слово." @@ -2162,7 +2230,7 @@ msgid "Enter " msgstr "Зарегистрироваться" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Вы действительно уверены?" @@ -2640,74 +2708,61 @@ msgstr "Добавить разрешенные / запрещенные тег msgid "Add Allowed/Denied custom column values" msgstr "Добавить разрешенные / запрещенные значения индивидуальных столбцов" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Читать" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Прослушать в браузере" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Опубликованный" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Отметить как непрочитанное" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Отметить как прочитанное" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Отметить как непрочитанное" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Прочесть" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Поиск" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Описание:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Добавить на книжную полку" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Публичная)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Редактировать метаданные" @@ -2779,10 +2834,6 @@ msgstr "Введите доменное имя" msgid "Denied Domains (Blacklist)" msgstr "Запрещенные домены (черный список)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Далее" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Откройте файл .kobo / Kobo eReader.conf в текстовом редакторе и добавьте (или отредактируйте):" @@ -2805,11 +2856,16 @@ msgstr "Сервер электронной почты не настроен, о msgid "Create Issue" msgstr "Создать запись" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Дополнительный Настройки" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Вернуться на главную" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2899,7 +2955,7 @@ msgstr "Книги, упорядоченные по рейтингу" msgid "Books ordered by file formats" msgstr "Книги отсортированы по формату файла" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Полки" @@ -2908,60 +2964,37 @@ msgstr "Полки" msgid "Books organized in shelves" msgstr "Книги организованы на полках" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Главная" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Включить навигацию" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Поиск в библиотеке" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Простой" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Учетная запись" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Выход" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Загружается..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Ошибка" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Загрузка завершена, обработка, пожалуйста, подождите..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Настройки" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Пожалуйста не обновляйте страницу" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Просмотр" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "О программе" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Предыдущий" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Подробнее о книге" @@ -3077,7 +3110,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Книга" @@ -3086,36 +3119,81 @@ msgstr "Книга" msgid "Calibre-Web eBook Catalog" msgstr "Каталог электронных книг Caliber-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "PDF reader" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Выберите имя пользователя" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Светлая" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Темная" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Назад" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Обновить размещение текста при открытии боковой панели." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Удалить" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Ожидание" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Вертикально" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Прочесть" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3297,10 +3375,6 @@ msgstr "Срок действия ссылки истекает через 10 м msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Результаты не найдены" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Выражение для поиска:" @@ -3317,11 +3391,11 @@ msgstr "Опубликовано от" msgid "Published Date To" msgstr "Опубликовано до" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3358,11 +3432,13 @@ msgstr "Рейтинг больше чем" msgid "Rating Below" msgstr "Рейтинг меньше чем" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3386,6 +3462,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Сделать книжную полку доступной для всех ?" @@ -3454,15 +3538,19 @@ msgstr "Прогресс" msgid "Run Time" msgstr "Время выполнения" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3470,6 +3558,10 @@ msgstr "" msgid "Reset user Password" msgstr "Сбросить пароль пользователя" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Показать книги на языках" diff --git a/cps/translations/sk/LC_MESSAGES/messages.mo b/cps/translations/sk/LC_MESSAGES/messages.mo index d18df97d..9536e295 100644 Binary files a/cps/translations/sk/LC_MESSAGES/messages.mo and b/cps/translations/sk/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/sk/LC_MESSAGES/messages.po b/cps/translations/sk/LC_MESSAGES/messages.po index 73aec973..f989879d 100644 --- a/cps/translations/sk/LC_MESSAGES/messages.po +++ b/cps/translations/sk/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2023-11-01 06:12+0100\n" "Last-Translator: Branislav Hanáček \n" "Language: sk_SK\n" @@ -16,497 +16,515 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Štatistika" -#: cps/admin.py:150 +#: cps/admin.py:151 msgid "Server restarted, please reload page." msgstr "Server bol reštartovaný, znovu načítajte stránku." -#: cps/admin.py:152 +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." msgstr "Vypínam server, zavrite prosím okno." -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "Spojenie s databázovo bolo úspešne znovu naviazané" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Neznámy príkaz" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Úspech! Knihy boli zaradená do zálohovania metadát, skontrolujte si Úlohy na kontrolu" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Neznámy" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Stránka správcu" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Základná konfigurácia" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Konfigurácia používateľského rozhrania" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Používateľom definovaný stĺpec č. %(column)d v Calibre databáze neexistuje" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Upraviť používateľov" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Všetko" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Používateľ sa nenašiel" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "Zmazaných {} používateľov" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Zobraziť všetko" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Chybne vytvorená žiadosť" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Meno hosťa nie je možné zmeniť" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Hosť nemôže mať túto rolu" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Nezostáva žiadny správca, nie je možné odobrať rolu správcu" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Hodnota musí byť pravda alebo nepravda" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Neplatná rola" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Hosť nemôže mať toto zobrazenie" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Neplatné zobrazenie" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Nastavenie jazyka pre hosťa sa určí automaticky a nie je možné ho nastaviť" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Nebolo poskytnuté platné nastavenie jazyka" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Nebolo poskytnuté platné jazykové nastavenie pre knihu" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parameter sa nenašiel" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Neplatný stĺpec na čítanie" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Neplatný stĺpec na obmedzenie" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Konfigurácia Calibre-Web bola aktualizovaná" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Naozaj chcete zmazať Kobo-žetón?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Naozaj chcete zmazať túto doménu?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Naozaj chcete zmazať tohot používateľa?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Naozaj chcete zmazať túto poličku?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Naozaj chcete zmeniť nastavenia jazykov pre vybraných používateľov?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Naozaj chcete zmeniť viditeľné jazkyky pre knihy pre vybraných používateľov?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Naozaj chcete zmeniť vybrané role pre vybraných používateľov?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Naozaj chcete zmeniť vybrané obmedzenia pre vybraných používateľov?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Naozaj chcete zmeniť vybrané obmedzenia viditeľnosti pre vybraných používateľov?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Naozaj chcete zmeniť synchronizačné správanie sa police pre vybraných používateľov?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Naozaj chcete zmenit umiestnenie pre knižnicu Calibre?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "Calilbre-Web vyhľadá aktualizované obálky a aktualizuje ich náhľady, môže to chvíľu trvať?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Naozaj chcete zmazať synchronizačnú databázu Calibre-Web aby ste vynútili úplnú synchronizáciu s vašou Kobo čítačkou?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Odmietnuť" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Povoliť" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "Zmazalo sa {} synchronizačných položiek" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Štítok sa nenašiel" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Neplatná akcia" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json nie je nakonfiguraovaný pre webovú aplikáciu" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Umiestnenie denníkového súboru je neplatné, zadajte prosím správnu cestu" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Umiestnenie súbor denníka prístupov nie je platné, zadajte prosím správnu cestu" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Zadajte prosím poskytovateľa LDAP, port, DN a identifikátor používateľa" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "Zadajte prosím konto a heslo pre LDAP službu" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "Zadajte prosím konto pre LDAP službu" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "Filter pre LDAP objekt skupiny potrebuje jeden \"%s\" formátový identifikátor" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "Filter pre LDAP objekt skupiny má nezhodu v zátvorkách" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "Filter pre LDAP objekt používateľa potrebuje jeden \"%s\" formátový identifikátor" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "Filter pre LDAP objekt používateľa má nezhodu v zátvorkách" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Filter pre LDAP členstvo používateľa potrebuje jeden \"%s\" formátový identifikátor" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Filter pre LDAP členstvo používateľa má nezhodu v zátvorkách" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CA-certifikát, certifikát alebo umiestnenie kľúča nie je platné. Zadajte prosím správnu cestu" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Pridať nového používateľa" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Upraviť nastavenie pre poštový server" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "Úspech! Gmail konto bolo verifikované." -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Chyba databázy: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Testovací e-mail bol zaradený na odoslanie na %(email)s, skontrolujte si výsledok v sekcii Úlohy" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Pri odosielaní testovacieho e-mailu sa vyskytla chyba: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Prosím, najskôr si nastavte e-mailovú adresu..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "Nastavenia poštového servera boli aktualizované" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "Upraviť nastavenia naplánovaných úloh" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "Bol uvedený neplatný čas spustenia" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "Bol uvedený neplatný doba behu" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "Nastavenia naplánovaných úloh boli aktualizované" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Vyskytla sa neočakávaná chyba. Prosím, skúste to opäť neskôr." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "Nie je možné zapisovať do databázy nastavení" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Upraviť používateľa %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "Úspech! Heslo pre používateľa %(user)s bolo resetované" -#: cps/admin.py:1451 +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." msgstr "Nastavte prosím poštový server." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Prezerač denníkového súboru" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Žiadosť o aktualizačný balík" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Sťahuje sa aktualizačný balík" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Rozbaľuje sa aktualizačný balík" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Nahradzujú sa súbory" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Spojenia s databázou sú uzavreté" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Zastavuje sa server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Aktualizácia dokončená, stlačte prosím OK a znovu načítajte stránku" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Aktualizácia zlyhala:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP chyba" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Chyba spojenia" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Časový limit pre naviazanie spojenia vypršal" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Všeobecná chyba" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "Súbor aktualizácie nebolo možné uložiť do dočasného adresára" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "Nebolo možné nahradiť súbory počas aktualizácie" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "Extrakcia minimálne jedného LDAP používateľa zlyhala" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Vytvorenie minimálne jedného LDAP používateľa zlyhalo" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Chyba: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Chyba: Odpoveď LDAP servera neobsahuje žiadneho používateľa" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Minimálne jeden LDAP používateľ sa nenachádza v databáze" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "Používateľ bol naimportovaný" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "Umiestnenie databáze nie je platné, zadajte prosím správnu cestu" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "Do databázy nie je možné zapisovať" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Umiestnenie súboru s kľúčmi nie je platné, zadajte prosím správnu cestu" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Umiestnenie súboru s certifikátmi nie je platné, zadajte prosím správnu cestu" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "Heslo musí byť dlhé 1 až 40 znakov" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Nastavenia databáze boli aktualizované" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Konfigurácia databázy" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Vyplňte prosím všetky polia." -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-mail nie je z platnej domény" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Pridať nového používateľa" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Používateľ '%(user)s' bol vytvorený" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Účet s týmto menom alebo e-mailovou adresou už existuje." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Používateľ '%(nick)s' bol zmazaný" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Nie je možné zmazať používateľa Hosť" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Nezostáva žiadny používateľ, nie je možné odobrať rolu správcu" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "E-mailová adresa nemôže byť prázdna a musí byť platná" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Používateľ '%(nick)s' bol aktualizovaný" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Hľadať" + #: cps/converter.py:31 msgid "not installed" msgstr "nie je naištalované" @@ -515,128 +533,123 @@ msgstr "nie je naištalované" msgid "Execution permissions missing" msgstr "Chýba právo na vykonanie" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Používateľom definovaný stĺpec č. %(column)d v Calibre databáze neexistuje" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Žiadne" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Vybraná kniha nie je dostupná. Súbor neexistuje alebo sa k nemu nedá pristupovať" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "Používateľ nemá práva nahrať obálku knihy" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Identifikátory nerozlišujú malé a veľké písmená, prepisuje sa starý identifikátor" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadáta boli úspešne aktualizované" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "Chyba pri úprave knihy: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Súbor %(file)s bol nahraný" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Chýba zdrojový alebo cieľový formát pre prevod" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Kniha bola úspešne zaradená na prevod do %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Vyskytla sa chyba pri prevode tejto knihy: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Nahrávaná kniha pravdepodobne existuje v knižnici, zvážte zmenu pred nahraním novej: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Vybraná kniha nie je dostupná. Súbor neexistuje alebo sa k nemu nedá pristupovať" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Používateľ nemá práva nahrať obálku knihy" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Identifikátory nerozlišujú malé a veľké písmená, prepisuje sa starý identifikátor" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "'%(langname)s' nie je platný jazyk" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadáta boli úspešne aktualizované" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Chyba pri úprave knihy: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Nahrávaná kniha pravdepodobne existuje v knižnici, zvážte zmenu pred nahraním novej: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Prípona súboru '%(ext)s' nemôže byť nahraná na tento server" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Prípona súboru '%(ext)s' nemôže byť nahraná na tento server" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Súbor ktorý sa má nahrať musí mať príponu" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Súbor %(filename)s nebolo možné uložiť do dočasného adresára" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Zlyhal presun súboru obalky %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Kniha s formátom bola úspešne zmazaná" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Kniha bola úspešne zmazaná" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "Nemáte oprávnenia na mazanie kníh" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "upraviť metadáta" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s nie je platné číslo, preskakuje sa" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "Používateľ nemá práva na nahranie dodatočných súborových formátov" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Nebolo možné vytvoriť cestu %(path)s (Povolenie zamietnuté)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Zlyhalo uloženie súboru: %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Súborový formát %(ext)s bol pridaný k %(book)s" @@ -649,470 +662,463 @@ msgstr "Nastavenie Google Drive nie je dokončené, skúste deaktivovať a znovu msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Doména spätného volania nie je verifikovaná, prosím, vykonajte kroky na verifikáciu domény v konzole Google vývojára" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "Formát %(format)s sa nenašiel pre knihu s ID: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s sa nenašiel na Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s sa nenašiel: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "Poslať do čítačky" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "Tento e-mail bol odoslaný skrz Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "Testovací e-mail Calibre-Web" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "Testovací e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Začnite s aplikáciu Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "Registračný e-mail pre používateľa: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Previesť %(orig)s na %(format)s a odoslať do čítačky" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, python-format msgid "Send %(format)s to eReader" msgstr "Odoslať %(format)s do čítačky" -#: cps/helper.py:227 +#: cps/helper.py:230 #, python-format msgid "%(book)s send to eReader" msgstr "%(book)s odoslaná do čítačky" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Požadovaný súbor sa nedá čítať. Možne zlé oprávnenia?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "Status čítania nie je možné nastaviť: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Mazanie zložky pre knihu %(id)s zlyhalo, cesta má vnorené adresáre: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Mazanie knihy %(id)s zlyhalo: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Mazanie knihy %(id)s iba z databázy, cesta ku knihe v databáze nie je platná: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Premenovanie autora z: '%(src)s' na '%(dest)s' zlyhalo s chybou: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Súbor %(file)s sa nenašiel na Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Zmena názvu knihy z: '%(src)s' na '%(dest)s' zlyhalo s chybou: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Cesta ku knihe %(path)s sa nenašla na Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "Pre túto poštovú adresu sa našiel existujúci účet" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Toto meno používateľa sa už používa" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "Neplatný formát poštovej adresy" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "Heslo nedodržiava pravidlá validácie" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "Python modul 'advocate' nie je nainštalovaný ale je potrebný pre nahrávanie obálok kníh" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Chyba pri sťahovaní obálky knihy" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Chyba formátu obálky knihy" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "Nemáte povolené pristupovať na lokálneho hostiteľa alebo lokálnu sieť na pre nahrávanie obálok kníh" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Vytváranie cesty k obálke knihy zlyhalo" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Súbor obálky knihy nie je platný súbor s obrázkom alebo nie je uložený" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Ako súbor obálky knihy sú podporované iba súbory jpg/jpeg/png/webp/bmp" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "Neplatný obsah súboru obalky knihy" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Ako súbor obálky knihy sú podporované iba súbory jpg/jpeg" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" msgstr "Obálka knihy" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "Binárny súbor pre UnRar sa nenašiel" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "Chyba pri spustení UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "Do databázy nie je možné zapisovať" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Chýba právo na vykonanie" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Chyba pri spustení UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "Zaradiť všetky knihy na zálohovanie metadát" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Prosím, pristupujte k Calibre-Web z nie lokálneho hostiteľa aby ste získali platný API koncový bod pre Kobo zariadenie" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Nastavenie Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Zaregistrovať s %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Úspech! Teraz ste prihlásený ako: %(nickname)s" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Spojenie na %(oauth)s bolo úspešné" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Prihlásenie zlyhalo, s OAuth účtom nie je spojený žiadny používateľ" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Odpojenie z %(oauth)s bolo úspešné" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Odpojenie z %(oauth)s zlyhalo" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Nie je pripojený k %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Prihlásenie na GitHub zlyhalo." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Získanie používateľa z GitHub-u zlyhalo." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Prihlásenie na Google zlyhalo." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Získanie používateľa z Google zlyhalo." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Chyba GitHub OAuth, skúste to prosím neskôr." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Chyba GitHub OAuth: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Chyba Google OAuth, skúste to prosím neskôr." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Chyba Google OAuth: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} Hviezdičiek" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Prihlásiť sa" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Žetón sa nenašiel" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Žetón expiroval" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Úspech! Vráťte sa k vašemu zariadeniu" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Knihy" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Zobraziť nedávno čítané knihy" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Horúce knihy" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Zobraziť horúce knihy" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Stiahnuté knihy" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Zobraziť stiahnuté knihy" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Najlepšie hodnotené knihy" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Zobraziť najlepšie hodnotené knihy" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Prečítané knihy" -#: cps/render_template.py:64 +#: cps/render_template.py:63 msgid "Show Read and Unread" msgstr "Zobraziť prečítané a neprečítané" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Neprečítané knihy" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Zobraziť neprečítané" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Objaviť" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Zobraziť náhodné knihy" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategórie" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" msgstr "Zobraziť časť Kategórie" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Série" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" msgstr "Zobraziť časť Série" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Autori" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" msgstr "Zobraziť časť Autori" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Vydavatelia" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" msgstr "Zobraziť časť vydavatelia" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Jazyky" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" msgstr "Zobraziť časť Jazyky" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Hodnotenia" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" msgstr "Zobraziť časť Hodnotenia" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Súborové formáty" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" msgstr "Zobraziť časť Súborové formáty" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Archivované knihy" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "Zobraziť archivované knihy" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Zoznam kníh" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Zobraziť zoznam kníh" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Hľadať" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Vydané po " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Vydané pred " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Hodnotenie <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Hodnotenie >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Status čítania = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "Chyba pri čítaní používateľom definovaných stĺpcov, reštartujte prosím Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Rozšírené hľadanie" @@ -1167,7 +1173,7 @@ msgstr "Kniha bola odstránená z police: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Ľutujeme, ale nie ste oprávnený odstrániť knihu z tejto police" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Vytvoriť policu" @@ -1220,45 +1226,45 @@ msgstr "Verejná polica s názvom '%(title)s' už existuje." msgid "A private shelf with the name '%(title)s' already exists." msgstr "Súkromná polica s názvom '%(title)s' už existuje." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Polica: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Chyba pri otváraní police. Polica neexistuje alebo je neprístupná" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Úlohy" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Čakajúca" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Neúspešná" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Spustená" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Hotová" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Ukončená" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Zrušená" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Neznámy stav" @@ -1291,170 +1297,170 @@ msgstr "Je dostupná nová aktualizácia. Kliknite na tlačidlo dolu, ak chcete msgid "No release information available" msgstr "Nie je dostupná informácia o vydaní" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Objaviť (Náhodné knihy)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Horúce knihy (Najviac sťahované)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Knihy stiahnuté: %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Author: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Vydavateľ: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Série: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Hodnotenie: Žiadne" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Hodnotenie: %(rating)s hviezdičiek" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Súborový formát: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategória: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Jazyk: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Stiahnutia" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Zoznam hodnotení" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Zoznam súborových formátov" -#: cps/web.py:1249 +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." msgstr "Najskôr nastavte poštový server, prosím..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Úspech! Kniha bola zaradená na odoslanie do %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Vyskytla sa chyba pri posielaní knihy: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Nastavte vo vašom profile platnú e-mailovú adresu pre vašu čítačku." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "Počkajte, prosím minútku pred registráciou ďalšieho používateľa" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registrovať" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "Poštový server nie je nastavený, kontaktujte prosím správcu." -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "Poštový server nie je nastavený, kontaktujte prosím správcu." -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Vaša e-mailová adresa nie je povolená." -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Úspech! Potvrdzujúci e-mail bol odoslaný." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "Nie je možné aktivovať LDAP autentifikáciu" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "Počkajte, prosím minútku pred opätovným prihlásením" -#: cps/web.py:1400 +#: cps/web.py:1408 #, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "ste prihlásený ako: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "Záložné prihlásenie ako: '%(nickname)s', LDAP server je nedostupný alebo používateľ je neznámy" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "Nemôžem prihlásiť: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" msgstr "Nesprávne používateľské meno alebo heslo" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "Na vašu e-mailovú adresu bolo odoslané nové heslo" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "Vyskytla sa neočakávaná chyba. Prosím, skúste to opäť neskôr." -#: cps/web.py:1429 +#: cps/web.py:1437 msgid "Please enter valid username to reset password" msgstr "Na resetovanie hesla zadajte platné používateľské meno" -#: cps/web.py:1437 +#: cps/web.py:1445 #, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Teraz ste prihlásený ako: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Profil pre %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 msgid "Success! Profile Updated" msgstr "Úspech! Profil bol aktualizovaný" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Účet s touto e-mailovou adresou už existuje." @@ -1462,54 +1468,58 @@ msgstr "Účet s touto e-mailovou adresou už existuje." msgid "Found no valid gmail.json file with OAuth information" msgstr "Nenašiel sa súbor gmail.json s OAuth informáciou" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s bol odoslaný do čítačky" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Nástroj pre prevod ebook-convert %(tool)s sa nenašiel" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "Formát %(format)s sa nenachádza na disku" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "Prevádzač e-kníh zlyhal s neznámou chybou" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Prevádzač Kepubify zlyhal: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Prevádzaný súbor sa nenašiel alebo viac ako jeden súbor v zložke %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "Prevádzač e-kníh zlyhal: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre zlyhal s chybou: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Prevádzač e-kníh zlyhal: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "Previesť" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "Spojenie s databázou sa znovu naväzuje" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "E-mail" @@ -1517,30 +1527,26 @@ msgstr "E-mail" msgid "Backing up Metadata" msgstr "Zálohovať metadáta" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "Bolo vygenerovaných %(count)s náhľadov obálok kníh" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "Náhľad obálky knihy" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "Bolo vygenerovaných {0} náhľadov pre série" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "Vyrovnávacia pamäť pre obálky kníh sa vyprázdňuje" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Nahrať" @@ -1559,11 +1565,11 @@ msgstr "Meno používateľa" msgid "Email" msgstr "E-mail" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "Poslať na e-mail čítačky" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Správca" @@ -1573,8 +1579,8 @@ msgstr "Správca" msgid "Password" msgstr "Heslo" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Stiahnuť" @@ -1784,13 +1790,13 @@ msgid "OK" msgstr "V poriadku" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Zrušiť" @@ -1840,16 +1846,76 @@ msgstr "Triediť podľa dátumu vydania, najnovšie najskôr" msgid "Sort according to publishing date, oldest first" msgstr "Triediť podľa dátumu vydania, najstaršie najskôr" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "znížiť" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Viac od" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Kniha %(index)s z %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Jazyk" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Vydavateľ" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Vydané" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Popis:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Predchádzajúci" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Ďalší" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Nič sa nenašlo" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Domov" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Prehľadávať knižnicu" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Odhlásiť sa" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Zmazať knihu" @@ -1878,99 +1944,107 @@ msgstr "Previesť do:" msgid "Convert book" msgstr "Previesť knihu" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Nahráva sa..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Zatvoriť" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Chyba" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Nahrávanie ukončené, spracováva sa, počkajte prosím..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Formát nahrávania" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Názov knihy" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Autor" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Značka" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "ID série" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Dátum vydania" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Hodnotenie" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Popis" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identifikátory" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Typ identifikátora" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Hodnota identifikátora" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Odstrániť" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Pridať identifikátor" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Značka" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "ID série" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Hodnotenie" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Načítať obálku z URL (JPEG-obrázok sa stiahne a uloží v databáze)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Nahrať obálku knihy z miestneho disku" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Dátum vydania" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Vydavateľ" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Jazyk" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Áno" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nie" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Formát nahrávania" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Zobraziť knihu pri ukladaní" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Načítať metadáta" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -1978,37 +2052,31 @@ msgstr "Načítať metadáta" msgid "Save" msgstr "Uložiť" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Kľúčové slovo" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Vyhľadať kľúčové slovo" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Kliknite na oblálku aby ste načítali metadáta do formulára" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Načítava sa..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Zatvoriť" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Zdroj" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Chyba pri vyhľadávaní!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Nič sa nenaslo! Skúste, prosím iné kľúčové slovo." @@ -2115,7 +2183,7 @@ msgid "Enter " msgstr "Zadať " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Naozaj to chcete?" @@ -2589,74 +2657,61 @@ msgstr "Pridať Povolené/Zakázané značky" msgid "Add Allowed/Denied custom column values" msgstr "Pridať Povolené/Zakázané používateľom definované hodnoty stĺpca" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Čítať v prezerači" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Počúvať v prezerači" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Kniha %(index)s z %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Vydané" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Označiť ako neprečítané" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Označiť ako prečítané" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Označiť ako neprečítané" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Čítať" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Obnoviť z archívu" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Pridať do archívu" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Archivovaný" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Popis:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Pridať do police" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Verejné)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Upraviť metadáta" @@ -2726,10 +2781,6 @@ msgstr "Zadajte doménové meno" msgid "Denied Domains (Blacklist)" msgstr "Zakázané domény (Čierny zoznam)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Ďalší" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Otvoriť súbor .kobo/Kobo/Kobo eReader.conf v textovom editore a pridať (alebo upraviť):" @@ -2750,11 +2801,16 @@ msgstr "Inštancia Calibre-Web nie je nastavené, kontaktujte prosím vašeho sp msgid "Create Issue" msgstr "Vytvoriť hlásenie problému" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Konfigurácia databázy" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Návrat domov" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "Odhlásiť používateľa" @@ -2844,7 +2900,7 @@ msgstr "Knihy usporiadané podľa hodnotenia" msgid "Books ordered by file formats" msgstr "Knihy usporiadané podľa súborových formátov" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Police" @@ -2853,60 +2909,37 @@ msgstr "Police" msgid "Books organized in shelves" msgstr "Knihy organizované v policiach" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Domov" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Prepnúť navigáciu" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Prehľadávať knižnicu" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Jednoduchý" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Účet" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Odhlásiť sa" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Nahráva sa..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Chyba" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Nahrávanie ukončené, spracováva sa, počkajte prosím..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Nastavenia" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Prosím, neskúšajte znovu načítať stránku" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Prechádzať" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "O programe" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Predchádzajúci" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Detailu o knihe" @@ -3022,7 +3055,7 @@ msgstr "Rodičovský adresár" msgid "Select" msgstr "Vybrať" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3030,34 +3063,80 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Katalóg e-kníh Calibre-Web" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "čítačka epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Vyberte si meno používateľa" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Svetlé" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Tmavé" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "Sépia" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "Čierne" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Preformátovať text keď sú otvorené bočné panely." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "Veľkosť písma" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Zmazať" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Čakajúca" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Vertikálne" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Čítať" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Neplatný stĺpec na čítanie" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Čítačka komiksov" @@ -3234,10 +3313,6 @@ msgstr "Toto verifikačné prepojenie expiruje za 10 minút." msgid "Generate Series Cover Thumbnails" msgstr "Vygenerovať náhľady obálok kníh pre série" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Nič sa nenašlo" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Vyhľadať výraz:" @@ -3254,11 +3329,11 @@ msgstr "Dátum vydania od" msgid "Published Date To" msgstr "Dátum vydania do" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3294,11 +3369,13 @@ msgstr "Hodnotenie lepšie ako" msgid "Rating Below" msgstr "Hodnotenie horšie ako" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Od:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Do:" @@ -3322,6 +3399,16 @@ msgstr "Znemožniť zmenu poradia" msgid "Enable Change order" msgstr "Povoloť zmenu poradia" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Triediť podľa dátumu knihy, najnovšie najskôr" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Triediť podľa dátumu knihy, najstaršie najskôr" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Zdieľať s kýmkoľvek" @@ -3390,15 +3477,20 @@ msgstr "Pokrok" msgid "Run Time" msgstr "Čas spustenia" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Zlúčiť" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Akcie" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "Táto úloha bude zrušená. Akýkoľvek pokrok vykonaný v tejto úlohe bude uložený." -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "Ak je toto naplánovaná úloha, bude znovu spustená v najbližšom naplánovanom čase." @@ -3406,6 +3498,10 @@ msgstr "Ak je toto naplánovaná úloha, bude znovu spustená v najbližšom nap msgid "Reset user Password" msgstr "Resetovať heslo používateľa" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Jazyk kníh" diff --git a/cps/translations/sl/LC_MESSAGES/messages.mo b/cps/translations/sl/LC_MESSAGES/messages.mo new file mode 100644 index 00000000..e2216946 Binary files /dev/null and b/cps/translations/sl/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/sl/LC_MESSAGES/messages.po b/cps/translations/sl/LC_MESSAGES/messages.po new file mode 100644 index 00000000..11fe175c --- /dev/null +++ b/cps/translations/sl/LC_MESSAGES/messages.po @@ -0,0 +1,3616 @@ +# Slovenian translations for Calibre-Web. +# Copyright (C) 2024 Andrej Kralj +# This file is distributed under the same license as the Calibre-Web +# project. +# FIRST AUTHOR Andrej Kralj, 2024. +msgid "" +msgstr "" +"Project-Id-Version: Calibre-Web\n" +"Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-09-18 19:45+0200\n" +"Last-Translator: Andrej Kralj\n" +"Language: sl\n" +"Language-Team: \n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=utf-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Generated-By: Babel 2.15.0\n" + +#: cps/about.py:85 +msgid "Statistics" +msgstr "Statistika" + +#: cps/admin.py:151 +msgid "Server restarted, please reload page." +msgstr "Strežnik se je znova zagnal, prosimo, ponovno naložite stran." + +#: cps/admin.py:153 +msgid "Performing Server shutdown, please close window." +msgstr "Izvajanje zaustavitve strežnika, prosim zaprite okno." + +#: cps/admin.py:161 +msgid "Success! Database Reconnected" +msgstr "Podatkovna baza je uspešno ponovno povezana" + +#: cps/admin.py:164 +msgid "Unknown command" +msgstr "Neznan ukaz" + +#: cps/admin.py:175 +msgid "Success! Books queued for Metadata Backup, please check Tasks for result" +msgstr "Uspeh! Knjige so v čakalni vrsti za varnostno kopiranje metapodatkov, za rezultat preverite opravila" + +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 +msgid "Unknown" +msgstr "Neznano" + +#: cps/admin.py:233 +msgid "Admin page" +msgstr "Administrativna stran" + +#: cps/admin.py:253 +msgid "Basic Configuration" +msgstr "Osnovne nastavitve" + +#: cps/admin.py:291 +msgid "UI Configuration" +msgstr "Nastavitve uporabniškega vmesnika" + +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Stolpec po meri št. %(column)d ne obstaja v zbirki podatkov Calibre" + +#: cps/admin.py:333 cps/templates/admin.html:51 +msgid "Edit Users" +msgstr "Urejanje uporabnikov" + +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 +#: cps/templates/list.html:13 +msgid "All" +msgstr "Vse" + +#: cps/admin.py:401 cps/admin.py:1426 +msgid "User not found" +msgstr "Ne najdem uporabnika" + +#: cps/admin.py:415 +msgid "{} users deleted successfully" +msgstr "{} uporabnikov uspešno izbrisanih" + +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 +#: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 +msgid "Show All" +msgstr "Pokaži vse" + +#: cps/admin.py:459 cps/admin.py:465 +msgid "Malformed request" +msgstr "Napačno oblikovana zahteva" + +#: cps/admin.py:477 cps/admin.py:2069 +msgid "Guest Name can't be changed" +msgstr "Imena gosta ni mogoče spremeniti" + +#: cps/admin.py:489 +msgid "Guest can't have this role" +msgstr "Gost ne more imeti te vloge" + +#: cps/admin.py:501 cps/admin.py:2023 +msgid "No admin user remaining, can't remove admin role" +msgstr "Ni več nobenega admin uporabnika, ne morem odstraniti vloge admin" + +#: cps/admin.py:505 cps/admin.py:519 +msgid "Value has to be true or false" +msgstr "Vrednost mora biti true ali false" + +#: cps/admin.py:507 +msgid "Invalid role" +msgstr "Neveljavna vloga" + +#: cps/admin.py:511 +msgid "Guest can't have this view" +msgstr "Gost ne more imeti tega pogleda" + +#: cps/admin.py:521 +msgid "Invalid view" +msgstr "Nepravilen pogled" + +#: cps/admin.py:524 +msgid "Guest's Locale is determined automatically and can't be set" +msgstr "Lokalni jezik gosta se določi samodejno in ga ni mogoče nastaviti." + +#: cps/admin.py:528 +msgid "No Valid Locale Given" +msgstr "Ni navedenega veljavnega lokalnega jezika" + +#: cps/admin.py:539 +msgid "No Valid Book Language Given" +msgstr "Ni navedenega veljavnega jezika knjige" + +#: cps/admin.py:541 cps/editbooks.py:292 +msgid "Parameter not found" +msgstr "Parameter ni najden" + +#: cps/admin.py:578 +msgid "Invalid Read Column" +msgstr "Nepravilen stolpec za branje" + +#: cps/admin.py:584 +msgid "Invalid Restricted Column" +msgstr "Neveljavni stolpec za omejitev" + +#: cps/admin.py:604 cps/admin.py:1894 +msgid "Calibre-Web configuration updated" +msgstr "Posodobljena nastavitev Calibre-Web" + +#: cps/admin.py:616 +msgid "Do you really want to delete the Kobo Token?" +msgstr "Ali res želite izbrisati žeton Kobo?" + +#: cps/admin.py:618 +msgid "Do you really want to delete this domain?" +msgstr "Ali res želite izbrisati to domeno?" + +#: cps/admin.py:620 +msgid "Do you really want to delete this user?" +msgstr "Ali res želite izbrisati tega uporabnika?" + +#: cps/admin.py:622 +msgid "Are you sure you want to delete this shelf?" +msgstr "Ste prepričani, da želite izbrisati to polico?" + +#: cps/admin.py:624 +msgid "Are you sure you want to change locales of selected user(s)?" +msgstr "Ali ste prepričani, da želite spremeniti lokalne jezike izbranih uporabnikov?" + +#: cps/admin.py:626 +msgid "Are you sure you want to change visible book languages for selected user(s)?" +msgstr "Ste prepričani, da želite spremeniti vidne jezike knjig za izbrane uporabnike?" + +#: cps/admin.py:628 +msgid "Are you sure you want to change the selected role for the selected user(s)?" +msgstr "Ali ste prepričani, da želite spremeniti izbrano vlogo za izbranega(-e) uporabnika(-e)?" + +#: cps/admin.py:630 +msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" +msgstr "Ali ste prepričani, da želite spremeniti izbrane omejitve za izbranega(-e) uporabnika(-e)?" + +#: cps/admin.py:632 +msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" +msgstr "Ali ste prepričani, da želite spremeniti izbrane omejitve vidljivosti za izbranega(-e) uporabnika(-e)?" + +#: cps/admin.py:635 +msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" +msgstr "Ali ste prepričani, da želite spremeniti obnašanje sinhronizacije police za izbranega(-e) uporabnika(-e)?" + +#: cps/admin.py:637 +msgid "Are you sure you want to change Calibre library location?" +msgstr "Ste prepričani, da želite spremeniti lokacijo knjižnice Calibre?" + +#: cps/admin.py:639 +msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" +msgstr "Calibre-Web bo poiskal posodobljene naslovnice in posodobil sličice naslovnic, kar lahko traja nekaj časa." + +#: cps/admin.py:642 +msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" +msgstr "Ali ste prepričani, da želite izbrisati sinhronizacijsko podatkovno bazo Calibre-Web, da bi vsilili popolno sinhronizacijo z bralnikom Kobo?" + +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 +#: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 +#: cps/templates/user_table.html:58 +msgid "Deny" +msgstr "Onemogoči" + +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 +#: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 +#: cps/templates/user_table.html:61 +msgid "Allow" +msgstr "Omogoči" + +#: cps/admin.py:946 +msgid "{} sync entries deleted" +msgstr "{} izbrisanih vnosov za sinhronizacijo" + +#: cps/admin.py:987 +msgid "Tag not found" +msgstr "Oznaka ni bila najdena" + +#: cps/admin.py:1005 +msgid "Invalid Action" +msgstr "Neveljavno dejanje" + +#: cps/admin.py:1132 +msgid "client_secrets.json Is Not Configured For Web Application" +msgstr "Client_secrets.json ni nastavljen za spletno aplikacijo" + +#: cps/admin.py:1177 +msgid "Logfile Location is not Valid, Please Enter Correct Path" +msgstr "Lokacija dnevniške datoteke ni veljavna, vnesite pravilno pot" + +#: cps/admin.py:1183 +msgid "Access Logfile Location is not Valid, Please Enter Correct Path" +msgstr "Lokacija dnevniške datoteke dostopa ni veljavna, vnesite pravilno pot" + +#: cps/admin.py:1217 +msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" +msgstr "Vnesite ponudnika LDAP, vrata, DN in identifikator objekta uporabnika" + +#: cps/admin.py:1223 +msgid "Please Enter a LDAP Service Account and Password" +msgstr "Vnesite račun in geslo storitve LDAP" + +#: cps/admin.py:1226 +msgid "Please Enter a LDAP Service Account" +msgstr "Vnesite račun storitve LDAP" + +#: cps/admin.py:1231 +#, python-format +msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" +msgstr "Filter predmetov skupine LDAP mora imeti en identifikator oblike \"%s\"" + +#: cps/admin.py:1233 +msgid "LDAP Group Object Filter Has Unmatched Parenthesis" +msgstr "Filter predmeta skupine LDAP ima neusklajene oklepaje" + +#: cps/admin.py:1237 +#, python-format +msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" +msgstr "Filter uporabniškega objekta LDAP mora imeti en identifikator oblike \"%s\"" + +#: cps/admin.py:1239 +msgid "LDAP User Object Filter Has Unmatched Parenthesis" +msgstr "Filter objekta uporabnika LDAP ima neusklajene oklepaje" + +#: cps/admin.py:1246 +#, python-format +msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" +msgstr "Filter za uporabnike članov LDAP mora imeti en identifikator oblike \"%s\"" + +#: cps/admin.py:1248 +msgid "LDAP Member User Filter Has Unmatched Parenthesis" +msgstr "Filter uporabnika člana LDAP ima neusklajene oklepaje" + +#: cps/admin.py:1255 +msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" +msgstr "LDAP CACcertifikat, lokacija certifikata ali ključa ni veljavna, vnesite pravilno pot" + +#: cps/admin.py:1286 cps/templates/admin.html:53 +msgid "Add New User" +msgstr "Dodajanje novega uporabnika" + +#: cps/admin.py:1295 cps/templates/admin.html:100 +msgid "Edit Email Server Settings" +msgstr "Urejanje nastavitev e-poštnega strežnika" + +#: cps/admin.py:1314 +msgid "Success! Gmail Account Verified." +msgstr "Uspeh! Račun Gmail je potrjen." + +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 +#: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 +#, python-format +msgid "Oops! Database Error: %(error)s." +msgstr "Ups! Napaka podatkovne baze: %(error)s." + +#: cps/admin.py:1344 +#, python-format +msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" +msgstr "Testno e-poštno sporočilo je v čakalni vrsti za pošiljanje na %(email)s naslovov, za rezultat preverite opravila" + +#: cps/admin.py:1347 +#, python-format +msgid "There was an error sending the Test e-mail: %(res)s" +msgstr "Pri pošiljanju testnega e-poštnega sporočila je prišlo do napake: %(res)s" + +#: cps/admin.py:1349 +msgid "Please configure your e-mail address first..." +msgstr "Najprej nastavite vaš e-poštni naslov..." + +#: cps/admin.py:1351 +msgid "Email Server Settings updated" +msgstr "Posodobljene nastavitve e-poštnega strežnika" + +#: cps/admin.py:1374 cps/templates/admin.html:195 +msgid "Edit Scheduled Tasks Settings" +msgstr "Urejanje nastavitev načrtovanih opravil" + +#: cps/admin.py:1386 +msgid "Invalid start time for task specified" +msgstr "Nepravilen začetni čas za določeno nalogo" + +#: cps/admin.py:1391 +msgid "Invalid duration for task specified" +msgstr "Nepravilno trajanje za določeno nalogo" + +#: cps/admin.py:1401 +msgid "Scheduled tasks settings updated" +msgstr "Posodobljene nastavitve načrtovanih opravil" + +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 +msgid "Oops! An unknown error occurred. Please try again later." +msgstr "Ups! Zgodila se je neznana napaka. Prosimo, poskusite znova pozneje." + +#: cps/admin.py:1415 +msgid "Settings DB is not Writeable" +msgstr "DB nastavitev ni mogoče zapisati" + +#: cps/admin.py:1445 cps/admin.py:2085 +#, python-format +msgid "Edit User %(nick)s" +msgstr "Urejanje uporabnika %(nick)s" + +#: cps/admin.py:1457 +#, python-format +msgid "Success! Password for user %(user)s reset" +msgstr "Uspeh! Ponastavitev gesla za uporabnika %(user)s" + +#: cps/admin.py:1463 +msgid "Oops! Please configure the SMTP mail settings." +msgstr "Ups! Nastavite nastavitve pošte SMTP." + +#: cps/admin.py:1474 +msgid "Logfile viewer" +msgstr "Pregledovalnik dnevniške datoteke" + +#: cps/admin.py:1540 +msgid "Requesting update package" +msgstr "Zahteva za paket posodobitev" + +#: cps/admin.py:1541 +msgid "Downloading update package" +msgstr "Prenos paketa posodobitev" + +#: cps/admin.py:1542 +msgid "Unzipping update package" +msgstr "Razpakiranje paketa posodobitev" + +#: cps/admin.py:1543 +msgid "Replacing files" +msgstr "Zamenjava datotek" + +#: cps/admin.py:1544 +msgid "Database connections are closed" +msgstr "Povezave do zbirke podatkov so zaprte" + +#: cps/admin.py:1545 +msgid "Stopping server" +msgstr "Ustavitev strežnika" + +#: cps/admin.py:1546 +msgid "Update finished, please press okay and reload page" +msgstr "Posodobitev je končana, pritisnite OK in ponovno naložite stran" + +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 +msgid "Update failed:" +msgstr "Posodobitev ni uspela:" + +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +msgid "HTTP Error" +msgstr "Napaka HTTP" + +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 +msgid "Connection error" +msgstr "Napaka povezave" + +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 +msgid "Timeout while establishing connection" +msgstr "Potek časa pri vzpostavljanju povezave" + +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 +msgid "General error" +msgstr "Splošna napaka" + +#: cps/admin.py:1551 +msgid "Update file could not be saved in temp dir" +msgstr "Datoteke posodobitve ni bilo mogoče shraniti v začasno mapo" + +#: cps/admin.py:1552 +msgid "Files could not be replaced during update" +msgstr "Datotek med posodabljanjem ni bilo mogoče zamenjati" + +#: cps/admin.py:1576 +msgid "Failed to extract at least One LDAP User" +msgstr "Nisem uspel izpisati vsaj enega uporabnika LDAP" + +#: cps/admin.py:1621 +msgid "Failed to Create at Least One LDAP User" +msgstr "Nisem uspel ustvariti vsaj enega uporabnika LDAP" + +#: cps/admin.py:1634 +#, python-format +msgid "Error: %(ldaperror)s" +msgstr "Napaka: %(ldaperror)s" + +#: cps/admin.py:1638 +msgid "Error: No user returned in response of LDAP server" +msgstr "Napaka: V odzivu strežnika LDAP ni vrnjenega nobenega uporabnika" + +#: cps/admin.py:1674 +msgid "At Least One LDAP User Not Found in Database" +msgstr "V podatkovni zbirki ni najden vsaj en uporabnik LDAP" + +#: cps/admin.py:1676 +msgid "{} User Successfully Imported" +msgstr "{} Uporabnik je bil uspešno uvožen" + +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "Pot do knjig ni veljavna" + +#: cps/admin.py:1740 +msgid "DB Location is not Valid, Please Enter Correct Path" +msgstr "Lokacija do DB ni veljavna, vnesite pravilno pot" + +#: cps/admin.py:1768 +msgid "DB is not Writeable" +msgstr "V DB ni mogoče zapisati" + +#: cps/admin.py:1782 +msgid "Keyfile Location is not Valid, Please Enter Correct Path" +msgstr "Lokacija datoteka Keyfile ni veljavna, vnesite pravilno pot" + +#: cps/admin.py:1786 +msgid "Certfile Location is not Valid, Please Enter Correct Path" +msgstr "Lokacija datoteke Certfile ni veljavna, vnesite pravilno pot" + +#: cps/admin.py:1863 +msgid "Password length has to be between 1 and 40" +msgstr "Dolžina gesla mora biti med 1 in 40" + +#: cps/admin.py:1917 +msgid "Database Settings updated" +msgstr "Posodobljene nastavitve zbirke podatkov" + +#: cps/admin.py:1925 +msgid "Database Configuration" +msgstr "Nastavitev zbirke podatkov" + +#: cps/admin.py:1940 cps/web.py:1299 +msgid "Oops! Please complete all fields." +msgstr "Ups! Izpolnite vsa polja." + +#: cps/admin.py:1949 +msgid "E-mail is not from valid domain" +msgstr "E-pošta ni iz veljavne domene" + +#: cps/admin.py:1955 +msgid "Add new user" +msgstr "Dodajanje novega uporabnika" + +#: cps/admin.py:1966 +#, python-format +msgid "User '%(user)s' created" +msgstr "Ustvarjen uporabnik '%(user)s'" + +#: cps/admin.py:1972 +msgid "Oops! An account already exists for this Email. or name." +msgstr "Ups! Za to e-pošto že obstaja račun ali ime." + +#: cps/admin.py:2002 +#, python-format +msgid "User '%(nick)s' deleted" +msgstr "Uporabnik '%(nick)s' je izbrisan" + +#: cps/admin.py:2005 +msgid "Can't delete Guest User" +msgstr "Ne morem izbrisati uporabnika gosta" + +#: cps/admin.py:2008 +msgid "No admin user remaining, can't delete user" +msgstr "Ni preostalega uporabnika administratorja, uporabnika ni mogoče izbrisati" + +#: cps/admin.py:2063 cps/web.py:1484 +msgid "Email can't be empty and has to be a valid Email" +msgstr "E-pošta ne sme biti prazna in mora biti veljavna." + +#: cps/admin.py:2089 +#, python-format +msgid "User '%(nick)s' updated" +msgstr "Uporabnik '%(nick)s' je posodobljen" + +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Iskanje" + +#: cps/converter.py:31 +msgid "not installed" +msgstr "ni nameščen" + +#: cps/converter.py:32 +msgid "Execution permissions missing" +msgstr "Manjkajo dovoljenja za izvajanje" + +#: cps/db.py:1038 cps/templates/config_edit.html:203 +#: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 +msgid "None" +msgstr "Nobeno" + +#: cps/editbooks.py:154 +#, python-format +msgid "File %(file)s uploaded" +msgstr "Prenesena datoteka %(file)s" + +#: cps/editbooks.py:183 +msgid "Source or destination format for conversion missing" +msgstr "Manjka izvorni ali ciljni format za pretvorbo" + +#: cps/editbooks.py:191 +#, python-format +msgid "Book successfully queued for converting to %(book_format)s" +msgstr "Knjiga je uspešno dodana v čakalno vrsto za pretvorbo v %(book_format)s" + +#: cps/editbooks.py:195 +#, python-format +msgid "There was an error converting this book: %(res)s" +msgstr "Pri pretvorbi te knjige je prišlo do napake: %(res)s" + +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Ups! Izbrana knjiga ni na voljo. Datoteka ne obstaja ali ni dostopna" + +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "Uporabnik nima pravice do nalaganja naslovnice" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Identifikatorji niso občutljivi na velikost črk, prepisovanje starega identifikatorja" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 +#, python-format +msgid "'%(langname)s' is not a valid language" +msgstr "'%(langname)s' ni veljaven jezik" + +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metapodatki so bili uspešno posodobljeni" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "Napaka pri urejanju knjige: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Naložena knjiga verjetno obstaja v knjižnici, preden naložite novo, razmislite o spremembi:" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 +msgid "File type isn't allowed to be uploaded to this server" +msgstr "Vrsta datoteke ni dovoljena za nalaganje v ta strežnik" + +#: cps/editbooks.py:761 cps/editbooks.py:1213 +#, python-format +msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" +msgstr "Datoteke s končnico '%(ext)s' ni dovoljeno naložiti na ta strežnik" + +#: cps/editbooks.py:765 cps/editbooks.py:1218 +msgid "File to be uploaded must have an extension" +msgstr "Datoteka, ki jo želite naložiti, mora imeti končnico" + +#: cps/editbooks.py:773 +#, python-format +msgid "File %(filename)s could not saved to temp dir" +msgstr "Datoteke %(filename)s ni bilo mogoče shraniti v začasno mapo" + +#: cps/editbooks.py:793 +#, python-format +msgid "Failed to Move Cover File %(file)s: %(error)s" +msgstr "Nisem uspel premakniti naslovne datoteke %(file)s: %(error)s" + +#: cps/editbooks.py:850 cps/editbooks.py:852 +msgid "Book Format Successfully Deleted" +msgstr "Uspešno izbrisana vrsta knjige" + +#: cps/editbooks.py:859 cps/editbooks.py:861 +msgid "Book Successfully Deleted" +msgstr "Knjiga je bila uspešno izbrisana" + +#: cps/editbooks.py:913 +msgid "You are missing permissions to delete books" +msgstr "Manjkajo vam dovoljenja za brisanje knjig" + +#: cps/editbooks.py:963 +msgid "edit metadata" +msgstr "urejanje metapodatkov" + +#: cps/editbooks.py:1016 +#, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "Številka serije: %(seriesindex)s ni veljavno število, preskočim" + +#: cps/editbooks.py:1207 +msgid "User has no rights to upload additional file formats" +msgstr "Uporabnik nima pravic za nalaganje dodatnih formatov datotek" + +#: cps/editbooks.py:1231 +#, python-format +msgid "Failed to create path %(path)s (Permission denied)." +msgstr "Nisem uspel ustvariti poti %(path)s (zavrnjeno dovoljenje)." + +#: cps/editbooks.py:1238 +#, python-format +msgid "Failed to store file %(file)s." +msgstr "Nisem uspel shraniti datoteke %(file)s." + +#: cps/editbooks.py:1263 +#, python-format +msgid "File format %(ext)s added to %(book)s" +msgstr "Oblina datotek %(ext)s je dodan v %(book)s" + +#: cps/gdrive.py:58 +msgid "Google Drive setup not completed, try to deactivate and activate Google Drive again" +msgstr "Nastavitev Google Drive ni dokončana, poskusite deaktivirati in znova aktivirati Google Drive" + +#: cps/gdrive.py:96 +msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" +msgstr "Povratna domena ni preverjena, sledite korakom za preverjanje domene v konzoli za razvijalce Google" + +#: cps/helper.py:87 +#, python-format +msgid "%(format)s format not found for book id: %(book)d" +msgstr "Vrsta %(format)s ni najdena za id knjige: %(book)d" + +#: cps/helper.py:94 cps/tasks/convert.py:93 +#, python-format +msgid "%(format)s not found on Google Drive: %(fn)s" +msgstr "%(format)s ni bil najden v storitvi Google Drive: %(fn)s" + +#: cps/helper.py:99 +#, python-format +msgid "%(format)s not found: %(fn)s" +msgstr "%(format)s ni najden: %(fn)s" + +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 +msgid "Send to eReader" +msgstr "Pošlji v e-bralnik" + +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 +msgid "This Email has been sent via Calibre-Web." +msgstr "To e-poštno sporočilo je bilo poslano prek programa Calibre-Web." + +#: cps/helper.py:123 +msgid "Calibre-Web Test Email" +msgstr "Testna e-pošta Calibre-Web" + +#: cps/helper.py:124 +msgid "Test Email" +msgstr "Testna e-pošta" + +#: cps/helper.py:141 +msgid "Get Started with Calibre-Web" +msgstr "Začnite s programom Calibre-Web" + +#: cps/helper.py:146 +#, python-format +msgid "Registration Email for user: %(name)s" +msgstr "Registracijski e-poštni naslov za uporabnika: %(name)s" + +#: cps/helper.py:157 cps/helper.py:163 +#, python-format +msgid "Convert %(orig)s to %(format)s and send to eReader" +msgstr "Pretvarjanje %(orig)s v %(format)s in pošiljanje v e-bralnik" + +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 +#, python-format +msgid "Send %(format)s to eReader" +msgstr "Pošlji %(format)s v e-bralnik" + +#: cps/helper.py:230 +#, python-format +msgid "%(book)s send to eReader" +msgstr "%(book)s pošljite v e-bralnik" + +#: cps/helper.py:237 +msgid "The requested file could not be read. Maybe wrong permissions?" +msgstr "Zahtevane datoteke ni bilo mogoče prebrati. Morda napačna dovoljenja?" + +#: cps/helper.py:352 +msgid "Read status could not set: {}" +msgstr "Stanja branja ni bilo mogoče nastaviti: {}" + +#: cps/helper.py:375 +#, python-format +msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" +msgstr "Brisanje knjižne mape za knjigo %(id)s ni bilo uspešno, pot ima podmape: %(path)s" + +#: cps/helper.py:381 +#, python-format +msgid "Deleting book %(id)s failed: %(message)s" +msgstr "Brisanje knjige %(id)s ni bilo uspešno: %(message)s" + +#: cps/helper.py:392 +#, python-format +msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" +msgstr "Brisanje knjige %(id)s samo iz zbirke podatkov, pot do knjige v zbirki podatkov ni veljavna: %(path)s" + +#: cps/helper.py:439 +#, python-format +msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" +msgstr "Preimenovanje avtorja iz: '%(src)s' v '%(dest)s' ni bilo uspešno z napako: %(error)s" + +#: cps/helper.py:507 cps/helper.py:516 +#, python-format +msgid "File %(file)s not found on Google Drive" +msgstr "Datoteke %(file)s ni mogoče najti v storitvi Google Drive" + +#: cps/helper.py:559 +#, python-format +msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" +msgstr "Preimenovanje naslova iz: '%(src)s' v '%(dest)s' ni bilo uspešno z napako: %(error)s" + +#: cps/helper.py:597 +#, python-format +msgid "Book path %(path)s not found on Google Drive" +msgstr "Pot do knjige %(path)s ni bila najdena v storitvi Google Drive" + +#: cps/helper.py:657 +msgid "Found an existing account for this Email address" +msgstr "Najden obstoječi račun za ta e-poštni naslov" + +#: cps/helper.py:665 +msgid "This username is already taken" +msgstr "To uporabniško ime je že zasedeno" + +#: cps/helper.py:679 +msgid "Invalid Email address format" +msgstr "Nepravilna oblike e-poštnega naslova" + +#: cps/helper.py:701 +msgid "Password doesn't comply with password validation rules" +msgstr "Geslo ni v skladu s pravili za preverjanje gesla" + +#: cps/helper.py:847 +msgid "Python module 'advocate' is not installed but is needed for cover uploads" +msgstr "Modul Python 'advocate' ni nameščen, vendar je potreben za nalaganje naslovnic" + +#: cps/helper.py:857 +msgid "Error Downloading Cover" +msgstr "Napaka pri nalaganju naslovnice" + +#: cps/helper.py:860 +msgid "Cover Format Error" +msgstr "Napaka vrste naslovnice" + +#: cps/helper.py:863 +msgid "You are not allowed to access localhost or the local network for cover uploads" +msgstr "Dostop do lokalnega gostitelja ali lokalnega omrežja za prenos naslovnice ni dovoljen." + +#: cps/helper.py:873 +msgid "Failed to create path for cover" +msgstr "Nisem uspel ustvariti poti za naslovnice" + +#: cps/helper.py:889 +msgid "Cover-file is not a valid image file, or could not be stored" +msgstr "Datoteka naslovnice ni veljavna slikovna datoteka ali je ni bilo mogoče shraniti" + +#: cps/helper.py:900 +msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" +msgstr "Za datoteke naslovnic so podprte samo datoteke jpg/jpeg/png/webp/bmp" + +#: cps/helper.py:912 +msgid "Invalid cover file content" +msgstr "Nepravilna vsebina datoteke z naslovnico" + +#: cps/helper.py:916 +msgid "Only jpg/jpeg files are supported as coverfile" +msgstr "Samo datoteke jpg/jpeg so podprte kot datoteke naslovnic" + +#: cps/helper.py:989 cps/helper.py:1149 +msgid "Cover" +msgstr "Naslovnica" + +#: cps/helper.py:1006 +msgid "UnRar binary file not found" +msgstr "Binarna datoteka UnRar ni bila najdena" + +#: cps/helper.py:1017 +msgid "Error executing UnRar" +msgstr "Napaka pri izvajanju UnRar" + +#: cps/helper.py:1025 +msgid "Could not find the specified directory" +msgstr "Ni bilo mogoče najti določenega imenika" + +#: cps/helper.py:1028 +msgid "Please specify a directory, not a file" +msgstr "Navedite imenik in ne datoteke" + +#: cps/helper.py:1042 +msgid "Calibre binaries not viable" +msgstr "Binarne datoteke Calibre niso izvedljive" + +#: cps/helper.py:1051 +#, python-format +msgid "Missing calibre binaries: %(missing)s" +msgstr "Manjkajoče binarne datoteke Calibre: %(missing)s" + +#: cps/helper.py:1053 +#, python-format +msgid "Missing executable permissions: %(missing)s" +msgstr "Manjkajoča dovoljenja za izvršilni program: %(missing)s" + +#: cps/helper.py:1058 +msgid "Error executing Calibre" +msgstr "Napaka pri izvajanju programa Calibre" + +#: cps/helper.py:1151 cps/templates/admin.html:216 +msgid "Queue all books for metadata backup" +msgstr "Vse knjige v vrsti za varnostno kopiranje metapodatkov" + +#: cps/kobo_auth.py:92 +msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" +msgstr "Dostopajte do Calibre-Web z nelokalnega gostitelja, da pridobite veljavno api_endpoint za napravo kobo" + +#: cps/kobo_auth.py:118 +msgid "Kobo Setup" +msgstr "Nastavitev za Kobo" + +#: cps/oauth_bb.py:78 +#, python-format +msgid "Register with %(provider)s" +msgstr "Registracija pri %(provider)s" + +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 +#, python-format +msgid "Success! You are now logged in as: %(nickname)s" +msgstr "Uspeh! Prijavljeni ste kot: %(nickname)s" + +#: cps/oauth_bb.py:149 +#, python-format +msgid "Link to %(oauth)s Succeeded" +msgstr "Povezava do %(oauth)s je bila uspešna" + +#: cps/oauth_bb.py:156 +msgid "Login failed, No User Linked With OAuth Account" +msgstr "Prijava ni uspela, z računom OAuth ni povezan noben uporabnik" + +#: cps/oauth_bb.py:198 +#, python-format +msgid "Unlink to %(oauth)s Succeeded" +msgstr "Odklop povezave z %(oauth)s je bil uspešen" + +#: cps/oauth_bb.py:203 +#, python-format +msgid "Unlink to %(oauth)s Failed" +msgstr "Odklop povezave z %(oauth)s ni bil uspešen" + +#: cps/oauth_bb.py:206 +#, python-format +msgid "Not Linked to %(oauth)s" +msgstr "Ni povezano z %(oauth)s" + +#: cps/oauth_bb.py:263 +msgid "Failed to log in with GitHub." +msgstr "Neuspešna prijava v GitHub." + +#: cps/oauth_bb.py:269 +msgid "Failed to fetch user info from GitHub." +msgstr "Neuspešna pridobitev informacij o uporabniku iz storitve GitHub." + +#: cps/oauth_bb.py:281 +msgid "Failed to log in with Google." +msgstr "Neuspešna prijava v Google." + +#: cps/oauth_bb.py:287 +msgid "Failed to fetch user info from Google." +msgstr "Neuspešna ridobititev informacij o uporabniku iz Googla." + +#: cps/oauth_bb.py:335 +msgid "GitHub Oauth error, please retry later." +msgstr "Napaka GitHub Oauth, prosimo, poskusite pozneje." + +#: cps/oauth_bb.py:338 +msgid "GitHub Oauth error: {}" +msgstr "Napaka GitHub Oauth: {}" + +#: cps/oauth_bb.py:359 +msgid "Google Oauth error, please retry later." +msgstr "Napaka Google Oauth, poskusite pozneje." + +#: cps/oauth_bb.py:362 +msgid "Google Oauth error: {}" +msgstr "Napaka Google Oauth: {}" + +#: cps/opds.py:299 +msgid "{} Stars" +msgstr "{} zvezdic" + +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 +msgid "Login" +msgstr "Prijava" + +#: cps/remotelogin.py:75 cps/remotelogin.py:109 +msgid "Token not found" +msgstr "ne najdem žetona" + +#: cps/remotelogin.py:84 cps/remotelogin.py:117 +msgid "Token has expired" +msgstr "Veljavnost žetona je potekla" + +#: cps/remotelogin.py:93 +msgid "Success! Please return to your device" +msgstr "Uspeh! Vrnite se v svojo napravo" + +#: cps/render_template.py:41 cps/web.py:424 +msgid "Books" +msgstr "Knjige" + +#: cps/render_template.py:43 +msgid "Show recent books" +msgstr "Prikaži nedavne knjige" + +#: cps/render_template.py:44 cps/templates/index.xml:27 +msgid "Hot Books" +msgstr "Najbolj vroče knjige" + +#: cps/render_template.py:46 +msgid "Show Hot Books" +msgstr "Prikaži vroče knjige" + +#: cps/render_template.py:48 cps/render_template.py:53 +msgid "Downloaded Books" +msgstr "Prenesene knjige" + +#: cps/render_template.py:50 cps/render_template.py:55 +#: cps/templates/user_table.html:167 +msgid "Show Downloaded Books" +msgstr "Prikaži prenesene knjige" + +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 +msgid "Top Rated Books" +msgstr "Najbolje ocenjene knjige" + +#: cps/render_template.py:60 cps/templates/user_table.html:161 +msgid "Show Top Rated Books" +msgstr "Prikaži najbolje ocenjene knjige" + +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 +msgid "Read Books" +msgstr "Prebrane knjige" + +#: cps/render_template.py:63 +msgid "Show Read and Unread" +msgstr "Prikaži prebrano in neprebrano" + +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 +msgid "Unread Books" +msgstr "Neprebrane knjige" + +#: cps/render_template.py:67 +msgid "Show unread" +msgstr "Prikaži neprebrane" + +#: cps/render_template.py:68 +msgid "Discover" +msgstr "Odkrivanje" + +#: cps/render_template.py:70 cps/templates/index.xml:58 +#: cps/templates/user_table.html:159 cps/templates/user_table.html:162 +msgid "Show Random Books" +msgstr "Prikaži naključne knjige" + +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 +msgid "Categories" +msgstr "Kategorije" + +#: cps/render_template.py:73 cps/templates/user_table.html:158 +msgid "Show Category Section" +msgstr "Prikaži oddelek kategorije" + +#: cps/render_template.py:74 cps/templates/book_edit.html:86 +#: cps/templates/book_table.html:68 cps/templates/index.xml:106 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 +msgid "Series" +msgstr "Serije" + +#: cps/render_template.py:76 cps/templates/user_table.html:157 +msgid "Show Series Section" +msgstr "Prikaži razdelek serij" + +#: cps/render_template.py:77 cps/templates/book_table.html:66 +#: cps/templates/index.xml:79 +msgid "Authors" +msgstr "Avtorji" + +#: cps/render_template.py:79 cps/templates/user_table.html:160 +msgid "Show Author Section" +msgstr "Prikaži razdelek avtorjev" + +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 +msgid "Publishers" +msgstr "Založniki" + +#: cps/render_template.py:83 cps/templates/user_table.html:163 +msgid "Show Publisher Section" +msgstr "Prikaži razdelek založnikov" + +#: cps/render_template.py:84 cps/templates/book_table.html:70 +#: cps/templates/index.xml:115 cps/templates/search_form.html:108 +#: cps/web.py:1113 +msgid "Languages" +msgstr "Jeziki" + +#: cps/render_template.py:87 cps/templates/user_table.html:155 +msgid "Show Language Section" +msgstr "Prikaži razdelek jezikov" + +#: cps/render_template.py:88 cps/templates/index.xml:124 +msgid "Ratings" +msgstr "Ocene" + +#: cps/render_template.py:90 cps/templates/user_table.html:164 +msgid "Show Ratings Section" +msgstr "Prikaži razdelek ocen" + +#: cps/render_template.py:91 cps/templates/index.xml:133 +msgid "File formats" +msgstr "Vrste datotek" + +#: cps/render_template.py:93 cps/templates/user_table.html:165 +msgid "Show File Formats Section" +msgstr "Prikaži razdelek vrste datotek" + +#: cps/render_template.py:95 cps/web.py:798 +msgid "Archived Books" +msgstr "Arhivirane knjige" + +#: cps/render_template.py:97 cps/templates/user_table.html:166 +msgid "Show Archived Books" +msgstr "Prikaži arhivirane knjige" + +#: cps/render_template.py:100 cps/web.py:829 +msgid "Books List" +msgstr "Seznam knjig" + +#: cps/render_template.py:102 cps/templates/user_table.html:168 +msgid "Show Books List" +msgstr "Prikaži seznam knjig" + +#: cps/search.py:201 +msgid "Published after " +msgstr "Objavljeno po" + +#: cps/search.py:208 +msgid "Published before " +msgstr "Objavljeno pred" + +#: cps/search.py:230 +#, python-format +msgid "Rating <= %(rating)s" +msgstr "Ocena <= %(rating)s" + +#: cps/search.py:232 +#, python-format +msgid "Rating >= %(rating)s" +msgstr "Ocena >= %(rating)s" + +#: cps/search.py:234 +#, python-format +msgid "Read Status = '%(status)s'" +msgstr "Prebrani status = '%(status)s'" + +#: cps/search.py:351 +msgid "Error on search for custom columns, please restart Calibre-Web" +msgstr "Napaka pri iskanju stolpcev po meri, ponovno zaženite Calibre-Web" + +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 +msgid "Advanced Search" +msgstr "Napredno iskanje" + +#: cps/shelf.py:49 cps/shelf.py:111 +msgid "Invalid shelf specified" +msgstr "Navedena nepravilna polica" + +#: cps/shelf.py:55 +msgid "Sorry you are not allowed to add a book to that shelf" +msgstr "Žal vam ni dovoljeno dodati knjige na to polico." + +#: cps/shelf.py:64 +#, python-format +msgid "Book is already part of the shelf: %(shelfname)s" +msgstr "Knjiga je že del police: %(shelfname)s" + +#: cps/shelf.py:77 +#, python-format +msgid "%(book_id)s is a invalid Book Id. Could not be added to Shelf" +msgstr "%(book_id)s je neveljaven ID knjige. Ni ga bilo mogoče dodati na polico" + +#: cps/shelf.py:97 +#, python-format +msgid "Book has been added to shelf: %(sname)s" +msgstr "Knjiga je bila dodana na polico: %(sname)s" + +#: cps/shelf.py:116 +msgid "You are not allowed to add a book to the shelf" +msgstr "Knjige ne smete dodati na polico" + +#: cps/shelf.py:134 +#, python-format +msgid "Books are already part of the shelf: %(name)s" +msgstr "Knjige so že del police: %(name)s" + +#: cps/shelf.py:146 +#, python-format +msgid "Books have been added to shelf: %(sname)s" +msgstr "Knjige so bile dodane na polico: %(sname)s" + +#: cps/shelf.py:153 +#, python-format +msgid "Could not add books to shelf: %(sname)s" +msgstr "Ni bilo mogoče dodati knjig na polico: %(sname)s" + +#: cps/shelf.py:199 +#, python-format +msgid "Book has been removed from shelf: %(sname)s" +msgstr "Knjiga je bila odstranjena s police: %(sname)s" + +#: cps/shelf.py:208 +msgid "Sorry you are not allowed to remove a book from this shelf" +msgstr "Žal vam ni dovoljeno odstraniti knjige s te police" + +#: cps/shelf.py:218 cps/templates/layout.html:160 +msgid "Create a Shelf" +msgstr "Ustvari polico" + +#: cps/shelf.py:226 +msgid "Sorry you are not allowed to edit this shelf" +msgstr "Žal vam ni dovoljeno urejati te police" + +#: cps/shelf.py:228 +msgid "Edit a shelf" +msgstr "Urejanje police" + +#: cps/shelf.py:237 +msgid "Error deleting Shelf" +msgstr "Napaka pri brisanju police" + +#: cps/shelf.py:239 +msgid "Shelf successfully deleted" +msgstr "Polica je bila uspešno izbrisana" + +#: cps/shelf.py:289 +#, python-format +msgid "Change order of Shelf: '%(name)s'" +msgstr "Spremeni vrstni red police: '%(name)s'" + +#: cps/shelf.py:324 +msgid "Sorry you are not allowed to create a public shelf" +msgstr "Žal vam ni dovoljeno ustvariti javne police" + +#: cps/shelf.py:341 +#, python-format +msgid "Shelf %(title)s created" +msgstr "Ustvarjena polica %(title)s" + +#: cps/shelf.py:344 +#, python-format +msgid "Shelf %(title)s changed" +msgstr "Polica %(title)s je spremenjena" + +#: cps/shelf.py:358 +msgid "There was an error" +msgstr "Prišlo je do napake" + +#: cps/shelf.py:380 +#, python-format +msgid "A public shelf with the name '%(title)s' already exists." +msgstr "Javna polica z imenom '%(title)s' že obstaja." + +#: cps/shelf.py:391 +#, python-format +msgid "A private shelf with the name '%(title)s' already exists." +msgstr "Zasebna polica z imenom '%(title)s' že obstaja." + +#: cps/shelf.py:481 +#, python-format +msgid "Shelf: '%(name)s'" +msgstr "Polica: '%(name)s'" + +#: cps/shelf.py:487 +msgid "Error opening shelf. Shelf does not exist or is not accessible" +msgstr "Napaka pri odpiranju police. Polica ne obstaja ali ni dostopna" + +#: cps/tasks_status.py:47 cps/templates/layout.html:91 +#: cps/templates/tasks.html:7 +msgid "Tasks" +msgstr "Opravila" + +#: cps/tasks_status.py:63 +msgid "Waiting" +msgstr "Na čakanju" + +#: cps/tasks_status.py:65 +msgid "Failed" +msgstr "Neuspešno" + +#: cps/tasks_status.py:67 +msgid "Started" +msgstr "Začeto" + +#: cps/tasks_status.py:69 +msgid "Finished" +msgstr "Končano" + +#: cps/tasks_status.py:71 +msgid "Ended" +msgstr "Končano" + +#: cps/tasks_status.py:73 +msgid "Cancelled" +msgstr "Preklicano" + +#: cps/tasks_status.py:75 +msgid "Unknown Status" +msgstr "Neznani status" + +#: cps/updater.py:433 cps/updater.py:444 cps/updater.py:545 cps/updater.py:560 +msgid "Unexpected data while reading update information" +msgstr "Nepričakovani podatki med branjem informacij o posodobitvi" + +#: cps/updater.py:440 cps/updater.py:552 +msgid "No update available. You already have the latest version installed" +msgstr "Posodobitev ni na voljo. Najnovejšo različico že imate nameščeno" + +#: cps/updater.py:458 +msgid "A new update is available. Click on the button below to update to the latest version." +msgstr "Na voljo je nova posodobitev. Klikni spodnji gumb za posodobitev na najnovejšo različico." + +#: cps/updater.py:476 +msgid "Could not fetch update information" +msgstr "Ni bilo mogoče pridobiti informacij o posodobitvi" + +#: cps/updater.py:486 +msgid "Click on the button below to update to the latest stable version." +msgstr "Klikni spodnji gumb za posodobitev na najnovejšo stabilno različico." + +#: cps/updater.py:495 cps/updater.py:509 cps/updater.py:520 +#, python-format +msgid "A new update is available. Click on the button below to update to version: %(version)s" +msgstr "Na voljo je nova posodobitev. Klikni spodnji gumb za posodobitev na različico: %(version)s" + +#: cps/updater.py:538 +msgid "No release information available" +msgstr "Informacije o izdaji niso na voljo" + +#: cps/templates/index.html:6 cps/web.py:451 +msgid "Discover (Random Books)" +msgstr "Odkrivanje (naključne knjige)" + +#: cps/web.py:487 +msgid "Hot Books (Most Downloaded)" +msgstr "Vroče knjige (največkrat prenesene)" + +#: cps/web.py:518 +#, python-format +msgid "Downloaded books by %(user)s" +msgstr "Prenesene knjige od %(user)s" + +#: cps/web.py:551 +#, python-format +msgid "Author: %(name)s" +msgstr "Avtor: %(name)s" + +#: cps/web.py:587 +#, python-format +msgid "Publisher: %(name)s" +msgstr "Založnik: %(name)s" + +#: cps/web.py:615 +#, python-format +msgid "Series: %(serie)s" +msgstr "Serija: %(serie)s" + +#: cps/web.py:629 +msgid "Rating: None" +msgstr "Ocena: brez" + +#: cps/web.py:638 +#, python-format +msgid "Rating: %(rating)s stars" +msgstr "Ocena: %(rating)s zvezdic" + +#: cps/web.py:669 +#, python-format +msgid "File format: %(format)s" +msgstr "Vrsta datoteke: %(format)s" + +#: cps/web.py:704 +#, python-format +msgid "Category: %(name)s" +msgstr "Kategorija: %(name)s" + +#: cps/web.py:733 +#, python-format +msgid "Language: %(name)s" +msgstr "Jezik: %(name)s" + +#: cps/templates/admin.html:16 cps/web.py:971 +msgid "Downloads" +msgstr "Prenosi" + +#: cps/web.py:1073 +msgid "Ratings list" +msgstr "Seznam ocen" + +#: cps/web.py:1100 +msgid "File formats list" +msgstr "Seznam vrste datotek" + +#: cps/web.py:1259 +msgid "Please configure the SMTP mail settings first..." +msgstr "Najprej nastavite nastavitve pošte SMTP..." + +#: cps/web.py:1265 +#, python-format +msgid "Success! Book queued for sending to %(eReadermail)s" +msgstr "Uspeh! Knjiga je v vrsti za pošiljanje v %(eReadermail)s" + +#: cps/web.py:1268 +#, python-format +msgid "Oops! There was an error sending book: %(res)s" +msgstr "Ups! Pri pošiljanju knjige je prišlo do napake: %(res)s" + +#: cps/web.py:1270 +msgid "Oops! Please update your profile with a valid eReader Email." +msgstr "Ups! Posodobite svoj profil z veljavnim e-poštnim naslovom eReaderja." + +#: cps/web.py:1286 +msgid "Please wait one minute to register next user" +msgstr "Počakajte eno minuto za registracijo naslednjega uporabnika" + +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 +msgid "Register" +msgstr "Registriraj" + +#: cps/web.py:1290 cps/web.py:1393 +msgid "Connection error to limiter backend, please contact your administrator" +msgstr "Napaka pri povezavi z zalednim strežnikom limiterja, obrnite se na skrbnika" + +#: cps/web.py:1295 cps/web.py:1342 +msgid "Oops! Email server is not configured, please contact your administrator." +msgstr "Ups! E-poštni strežnik ni nastavljen, obrnite se na skrbnika." + +#: cps/web.py:1328 +msgid "Oops! Your Email is not allowed." +msgstr "Ups! Vaša e-pošta ni dovoljena." + +#: cps/web.py:1331 +msgid "Success! Confirmation Email has been sent." +msgstr "Uspeh! Potrditveno e-poštno sporočilo je bilo poslano." + +#: cps/web.py:1376 cps/web.py:1399 +msgid "Cannot activate LDAP authentication" +msgstr "Ni mogoče aktivirati avtentikacije LDAP" + +#: cps/web.py:1389 +msgid "Please wait one minute before next login" +msgstr "Pred naslednjo prijavo počakajte eno minuto" + +#: cps/web.py:1408 +#, python-format +msgid "you are now logged in as: '%(nickname)s'" +msgstr "prijavljeni ste kot: '%(nickname)s'" + +#: cps/web.py:1415 +#, python-format +msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" +msgstr "Rezervna prijava kot: %(nickname)s, strežnik LDAP ni dosegljiv ali uporabnik ni znan" + +#: cps/web.py:1420 +#, python-format +msgid "Could not login: %(message)s" +msgstr "Ni se mogel prijaviti: %(message)s" + +#: cps/web.py:1424 cps/web.py:1449 +msgid "Wrong Username or Password" +msgstr "Napačno uporabniško ime ali geslo" + +#: cps/web.py:1431 +msgid "New Password was sent to your email address" +msgstr "Novo geslo je bilo poslano na vaš e-poštni naslov" + +#: cps/web.py:1435 +msgid "An unknown error occurred. Please try again later." +msgstr "Zgodila se je neznana napaka. Prosimo, poskusite znova pozneje." + +#: cps/web.py:1437 +msgid "Please enter valid username to reset password" +msgstr "Za ponastavitev gesla vnesite veljavno uporabniško ime" + +#: cps/web.py:1445 +#, python-format +msgid "You are now logged in as: '%(nickname)s'" +msgstr "Prijavljeni ste kot: '%(nickname)s'" + +#: cps/web.py:1510 cps/web.py:1560 +#, python-format +msgid "%(name)s's Profile" +msgstr "%(name)s profil" + +#: cps/web.py:1526 +msgid "Success! Profile Updated" +msgstr "Uspeh! Profil posodobljen" + +#: cps/web.py:1530 +msgid "Oops! An account already exists for this Email." +msgstr "Ups! Račun za to e-pošto že obstaja." + +#: cps/services/gmail.py:59 +msgid "Found no valid gmail.json file with OAuth information" +msgstr "Nisem našel veljavne datoteke gmail.json z informacijami OAuth" + +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "Izbriši vsebino začasne mape" + +#: cps/tasks/convert.py:112 +#, python-format +msgid "%(book)s send to E-Reader" +msgstr "%(book)s pošlji v e-bralnik" + +#: cps/tasks/convert.py:177 +#, python-format +msgid "Calibre ebook-convert %(tool)s not found" +msgstr "Calibre ebook-convert %(tool)s ni najden" + +#: cps/tasks/convert.py:211 +#, python-format +msgid "%(format)s format not found on disk" +msgstr "%(format)s vrsta ni najdena na disku" + +#: cps/tasks/convert.py:215 +msgid "Ebook converter failed with unknown error" +msgstr "Pretvornik e-knjig ni uspel z neznano napako" + +#: cps/tasks/convert.py:234 +#, python-format +msgid "Kepubify-converter failed: %(error)s" +msgstr "Kepubify-converter ni uspel: %(error)s" + +#: cps/tasks/convert.py:255 +#, python-format +msgid "Converted file not found or more than one file in folder %(folder)s" +msgstr "Pretvorjena datoteka ni bila najdena ali je v mapi %(folder)s več kot ena datoteka" + +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 +#, python-format +msgid "Calibre failed with error: %(error)s" +msgstr "Calibre ni uspel z napako: %(error)s" + +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "Ebook-converter ni uspel: %(error)s" + +#: cps/tasks/convert.py:345 +msgid "Convert" +msgstr "Pretvori" + +#: cps/tasks/database.py:26 +msgid "Reconnecting Calibre database" +msgstr "Ponovno se povezuejm na podatkovno zbirko Calibre" + +#: cps/tasks/mail.py:283 +msgid "E-mail" +msgstr "E-naslov" + +#: cps/tasks/metadata_backup.py:34 +msgid "Backing up Metadata" +msgstr "Varnostno kopiranje metapodatkov" + +#: cps/tasks/thumbnail.py:97 +#, python-format +msgid "Generated %(count)s cover thumbnails" +msgstr "Ustvarjene sličice naslovnic %(count)s" + +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 +msgid "Cover Thumbnails" +msgstr "Sličicah naslovnice" + +#: cps/tasks/thumbnail.py:294 +msgid "Generated {0} series thumbnails" +msgstr "Ustvarjene sličice serije {0}" + +#: cps/tasks/thumbnail.py:459 +msgid "Clearing cover thumbnail cache" +msgstr "Čiščenje predpomnilnika sličic naslovnice" + +#: cps/tasks/upload.py:39 cps/templates/admin.html:20 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 +msgid "Upload" +msgstr "Naloži" + +#: cps/templates/admin.html:9 +msgid "Users" +msgstr "Uporabniki" + +#: cps/templates/admin.html:13 cps/templates/login.html:9 +#: cps/templates/login.html:10 cps/templates/register.html:9 +#: cps/templates/user_edit.html:10 cps/templates/user_table.html:134 +msgid "Username" +msgstr "Uporabniško ime" + +#: cps/templates/admin.html:14 cps/templates/register.html:14 +#: cps/templates/user_edit.html:15 cps/templates/user_table.html:135 +msgid "Email" +msgstr "E-pošta" + +#: cps/templates/admin.html:15 +msgid "Send to eReader Email" +msgstr "Pošlji v e-pošto e-bralnika" + +#: cps/templates/admin.html:17 cps/templates/layout.html:94 +#: cps/templates/user_table.html:143 +msgid "Admin" +msgstr "Administator" + +#: cps/templates/admin.html:18 cps/templates/login.html:13 +#: cps/templates/login.html:14 cps/templates/user_edit.html:23 +msgid "Password" +msgstr "Geslo" + +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 +#: cps/templates/user_table.html:146 +msgid "Download" +msgstr "Prenesi" + +#: cps/templates/admin.html:23 +msgid "View Books" +msgstr "Oglejte si knjige" + +#: cps/templates/admin.html:24 cps/templates/user_table.html:131 +#: cps/templates/user_table.html:148 +msgid "Edit" +msgstr "Uredi" + +#: cps/templates/admin.html:25 cps/templates/book_edit.html:17 +#: cps/templates/book_table.html:100 cps/templates/modal_dialogs.html:63 +#: cps/templates/modal_dialogs.html:116 cps/templates/user_edit.html:67 +#: cps/templates/user_table.html:149 +msgid "Delete" +msgstr "Izbriši" + +#: cps/templates/admin.html:26 +msgid "Public Shelf" +msgstr "Javna polica" + +#: cps/templates/admin.html:55 +msgid "Import LDAP Users" +msgstr "Uvoz uporabnikov LDAP" + +#: cps/templates/admin.html:62 +msgid "Email Server Settings" +msgstr "Nastavitve e-poštnega strežnika" + +#: cps/templates/admin.html:67 cps/templates/email_edit.html:31 +msgid "SMTP Hostname" +msgstr "Ime gostitelja SMTP" + +#: cps/templates/admin.html:71 cps/templates/email_edit.html:35 +msgid "SMTP Port" +msgstr "Vrata SMTP" + +#: cps/templates/admin.html:75 cps/templates/email_edit.html:39 +msgid "Encryption" +msgstr "Šifriranje" + +#: cps/templates/admin.html:79 cps/templates/email_edit.html:47 +msgid "SMTP Login" +msgstr "Prijava v SMTP" + +#: cps/templates/admin.html:83 cps/templates/admin.html:94 +#: cps/templates/email_edit.html:55 +msgid "From Email" +msgstr "Iz e-poštnega naslova" + +#: cps/templates/admin.html:90 +msgid "Email Service" +msgstr "Storitev e-pošte" + +#: cps/templates/admin.html:91 +msgid "Gmail via Oauth2" +msgstr "Gmail prek Oauth2" + +#: cps/templates/admin.html:106 +msgid "Configuration" +msgstr "Nastavitev" + +#: cps/templates/admin.html:109 +msgid "Calibre Database Directory" +msgstr "Imenik zbirk podatkov Calibre" + +#: cps/templates/admin.html:113 cps/templates/config_edit.html:68 +msgid "Log Level" +msgstr "Raven dnevnika" + +#: cps/templates/admin.html:117 +msgid "Port" +msgstr "Vrata" + +#: cps/templates/admin.html:122 +msgid "External Port" +msgstr "Zunanja vrata" + +#: cps/templates/admin.html:129 cps/templates/config_view_edit.html:28 +msgid "Books per Page" +msgstr "Število knjig na stran" + +#: cps/templates/admin.html:133 +msgid "Uploads" +msgstr "Nalaganje" + +#: cps/templates/admin.html:137 +msgid "Anonymous Browsing" +msgstr "Anonimno brskanje" + +#: cps/templates/admin.html:141 +msgid "Public Registration" +msgstr "Javna registracija" + +#: cps/templates/admin.html:145 +msgid "Magic Link Remote Login" +msgstr "Prijava v oddaljeno prijavo Magic Link" + +#: cps/templates/admin.html:149 +msgid "Reverse Proxy Login" +msgstr "Prijava za povratni posredniški strežnik" + +#: cps/templates/admin.html:154 cps/templates/config_edit.html:172 +msgid "Reverse Proxy Header Name" +msgstr "Ime glave povratnega posredniškega strežnika" + +#: cps/templates/admin.html:159 +msgid "Edit Calibre Database Configuration" +msgstr "Urejanje nastavitev podatkovne zbirke Calibre" + +#: cps/templates/admin.html:160 +msgid "Edit Basic Configuration" +msgstr "Urejanje osnovnih nastavitev" + +#: cps/templates/admin.html:161 +msgid "Edit UI Configuration" +msgstr "Urejanje nastavitev uporabniškega vmesnika" + +#: cps/templates/admin.html:167 +msgid "Scheduled Tasks" +msgstr "Načrtovana opravila" + +#: cps/templates/admin.html:170 cps/templates/schedule_edit.html:12 +#: cps/templates/tasks.html:18 +msgid "Start Time" +msgstr "Čas začetka" + +#: cps/templates/admin.html:174 cps/templates/schedule_edit.html:20 +msgid "Maximum Duration" +msgstr "Najdaljše trajanje" + +#: cps/templates/admin.html:178 cps/templates/schedule_edit.html:29 +msgid "Generate Thumbnails" +msgstr "Ustvarjanje sličic" + +#: cps/templates/admin.html:182 +msgid "Generate series cover thumbnails" +msgstr "Ustvarjanje sličic naslovnic serij" + +#: cps/templates/admin.html:186 cps/templates/admin.html:208 +#: cps/templates/schedule_edit.html:37 +msgid "Reconnect Calibre Database" +msgstr "Ponovna povezava podatkovne zbirke Calibre" + +#: cps/templates/admin.html:190 cps/templates/schedule_edit.html:41 +msgid "Generate Metadata Backup Files" +msgstr "Ustvarjanje varnostnih datotek metapodatkov" + +#: cps/templates/admin.html:197 +msgid "Refresh Thumbnail Cache" +msgstr "Osvežitev predpomnilnika sličic" + +#: cps/templates/admin.html:203 +msgid "Administration" +msgstr "Administracija" + +#: cps/templates/admin.html:204 +msgid "Download Debug Package" +msgstr "Prenos paketa za odpravljanje napak" + +#: cps/templates/admin.html:205 +msgid "View Logs" +msgstr "Prikaži dnevnike" + +#: cps/templates/admin.html:211 +msgid "Restart" +msgstr "Ponovni zagon" + +#: cps/templates/admin.html:212 +msgid "Shutdown" +msgstr "Izklop" + +#: cps/templates/admin.html:221 +msgid "Version Information" +msgstr "Informacije o različici" + +#: cps/templates/admin.html:225 +msgid "Version" +msgstr "Različica" + +#: cps/templates/admin.html:226 +msgid "Details" +msgstr "Podrobnosti" + +#: cps/templates/admin.html:232 +msgid "Current Version" +msgstr "Trenutna različica" + +#: cps/templates/admin.html:239 +msgid "Check for Update" +msgstr "Preveri za posodobitev" + +#: cps/templates/admin.html:240 +msgid "Perform Update" +msgstr "Izvedite posodobitev" + +#: cps/templates/admin.html:253 +msgid "Are you sure you want to restart?" +msgstr "Ali ste prepričani, da želite znova zagnati računalnik?" + +#: cps/templates/admin.html:258 cps/templates/admin.html:272 +#: cps/templates/admin.html:292 cps/templates/config_db.html:82 +msgid "OK" +msgstr "V redu" + +#: cps/templates/admin.html:259 cps/templates/admin.html:273 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 +#: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 +#: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 +#: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 +#: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 +msgid "Cancel" +msgstr "Prekliči" + +#: cps/templates/admin.html:271 +msgid "Are you sure you want to shutdown?" +msgstr "Ste prepričani, da želite izklopiti?" + +#: cps/templates/admin.html:283 +msgid "Updating, please do not reload this page" +msgstr "Posodabljanje, prosimo, ne nalagajte te strani znova" + +#: cps/templates/author.html:15 +msgid "via" +msgstr "preko" + +#: cps/templates/author.html:23 +msgid "In Library" +msgstr "V knjižnici" + +#: cps/templates/author.html:26 cps/templates/index.html:74 +#: cps/templates/search.html:31 cps/templates/shelf.html:20 +msgid "Sort according to book date, newest first" +msgstr "Razvrstite glede na datum knjige, najprej najnovejši" + +#: cps/templates/author.html:27 cps/templates/index.html:75 +#: cps/templates/search.html:32 cps/templates/shelf.html:21 +msgid "Sort according to book date, oldest first" +msgstr "Razvrsti po datumu knjige, najprej najstarejši" + +#: cps/templates/author.html:28 cps/templates/index.html:76 +#: cps/templates/search.html:33 cps/templates/shelf.html:22 +msgid "Sort title in alphabetical order" +msgstr "Razvrsti naslov po abecednem vrstnem redu" + +#: cps/templates/author.html:29 cps/templates/index.html:77 +#: cps/templates/search.html:34 cps/templates/shelf.html:23 +msgid "Sort title in reverse alphabetical order" +msgstr "Razvrsti naslov v obratnem abecednem vrstnem redu" + +#: cps/templates/author.html:30 cps/templates/index.html:80 +#: cps/templates/search.html:37 cps/templates/shelf.html:26 +msgid "Sort according to publishing date, newest first" +msgstr "Razvrsti glede na datum objave, najprej najnovejše" + +#: cps/templates/author.html:31 cps/templates/index.html:81 +#: cps/templates/search.html:38 cps/templates/shelf.html:27 +msgid "Sort according to publishing date, oldest first" +msgstr "Razvrsti glede na datum objave, najprej najstarejši" + +#: cps/templates/author.html:56 cps/templates/author.html:113 +#: cps/templates/index.html:30 cps/templates/index.html:113 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 +msgid "reduce" +msgstr "zmanjšaj" + +#: cps/templates/author.html:97 +msgid "More by" +msgstr "Več po" + +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Knjiga %(index)s v %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Jezik" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Založnik" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Objavljeno" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Opis:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Prejšnji" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Naslednji" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Ni bilo najdenih rezultatov" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Domov" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Iskanje po knjižnici" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Odjava" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + +#: cps/templates/book_edit.html:11 +msgid "Delete Book" +msgstr "Izbriši knjigo" + +#: cps/templates/book_edit.html:14 +msgid "Delete formats:" +msgstr "Brisanje vrste:" + +#: cps/templates/book_edit.html:25 +msgid "Convert book format:" +msgstr "Pretvori vrsto knjige:" + +#: cps/templates/book_edit.html:30 +msgid "Convert from:" +msgstr "Pretvori iz:" + +#: cps/templates/book_edit.html:32 cps/templates/book_edit.html:39 +msgid "select an option" +msgstr "izberi možnost" + +#: cps/templates/book_edit.html:37 +msgid "Convert to:" +msgstr "Pretvori v:" + +#: cps/templates/book_edit.html:46 +msgid "Convert book" +msgstr "Pretvori knjigo" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Nalaganje..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Zapri" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Napaka" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Prenos opravljen, obdelujem, prosim počakajte..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Nalaganje vrsta" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 +msgid "Book Title" +msgstr "Naslov knjige" + +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 +msgid "Author" +msgstr "Avtor" + +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Oznake" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Številka serije" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Datum objave" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Ocena" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 +msgid "Description" +msgstr "Opis" + +#: cps/templates/book_edit.html:118 +msgid "Identifiers" +msgstr "Identifikatorji" + +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 +msgid "Identifier Type" +msgstr "Vrsta identifikatorja" + +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 +msgid "Identifier Value" +msgstr "Vrednost identifikatorja" + +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 +#: cps/templates/user_table.html:24 +msgid "Remove" +msgstr "Odstrani" + +#: cps/templates/book_edit.html:129 +msgid "Add Identifier" +msgstr "Dodaj identifikator" + +#: cps/templates/book_edit.html:133 +msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" +msgstr "Prenesi naslovnico iz naslova URL (JPEG - slika se prenese in shrani v zbirko podatkov)" + +#: cps/templates/book_edit.html:137 +msgid "Upload Cover from Local Disk" +msgstr "Prenesi naslovnico iz lokalnega diska" + +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 +msgid "Yes" +msgstr "Da" + +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 +msgid "No" +msgstr "Ne" + +#: cps/templates/book_edit.html:215 +msgid "View Book on Save" +msgstr "Oglej si knjigo po shranjevanju" + +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 +msgid "Fetch Metadata" +msgstr "Pridobivanje metapodatkov" + +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 +#: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 +#: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 +#: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 +#: cps/templates/user_edit.html:142 +msgid "Save" +msgstr "Shrani" + +#: cps/templates/book_edit.html:239 +msgid "Keyword" +msgstr "Ključna beseda" + +#: cps/templates/book_edit.html:240 +msgid "Search keyword" +msgstr "Ključna beseda za iskanje" + +#: cps/templates/book_edit.html:246 +msgid "Click the cover to load metadata to the form" +msgstr "Klikni na naslovnico, da se v obrazec naložijo metapodatki." + +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 +msgid "Loading..." +msgstr "Nalaganje..." + +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 +msgid "Source" +msgstr "Vir:" + +#: cps/templates/book_edit.html:293 +msgid "Search error!" +msgstr "Napaka pri iskanju!" + +#: cps/templates/book_edit.html:294 +msgid "No Result(s) found! Please try another keyword." +msgstr "Ni najden noben rezultat! Poskusite z drugo ključno besedo." + +#: cps/templates/book_table.html:12 cps/templates/book_table.html:69 +#: cps/templates/user_table.html:14 cps/templates/user_table.html:77 +#: cps/templates/user_table.html:100 +msgid "This Field is Required" +msgstr "To polje je obvezno" + +#: cps/templates/book_table.html:37 +msgid "Merge selected books" +msgstr "Združevanje izbranih knjig" + +#: cps/templates/book_table.html:38 cps/templates/user_table.html:124 +msgid "Remove Selections" +msgstr "Odstranjevanje izbire" + +#: cps/templates/book_table.html:41 +msgid "Exchange author and title" +msgstr "Zamenjava avtorja in naslova" + +#: cps/templates/book_table.html:47 +msgid "Update Title Sort automatically" +msgstr "Samodejna posodobitev razvrščanja po naslovu" + +#: cps/templates/book_table.html:51 +msgid "Update Author Sort automatically" +msgstr "Samodejna posodobitev razvrščanja po avtorju" + +#: cps/templates/book_table.html:63 cps/templates/book_table.html:69 +msgid "Enter Title" +msgstr "Vnesi naslov" + +#: cps/templates/book_table.html:63 cps/templates/config_view_edit.html:24 +#: cps/templates/shelf_edit.html:8 +msgid "Title" +msgstr "Naslov" + +#: cps/templates/book_table.html:64 +msgid "Enter Title Sort" +msgstr "Vnesi razvrščanje naslovov" + +#: cps/templates/book_table.html:64 +msgid "Title Sort" +msgstr "Razvrščanje naslovov" + +#: cps/templates/book_table.html:65 +msgid "Enter Author Sort" +msgstr "Vnesi razvrščanje avtorjev" + +#: cps/templates/book_table.html:65 +msgid "Author Sort" +msgstr "Razvrščanje avtorjev" + +#: cps/templates/book_table.html:66 +msgid "Enter Authors" +msgstr "Vnesi avtorja" + +#: cps/templates/book_table.html:67 +msgid "Enter Categories" +msgstr "Vnesi kategorijo" + +#: cps/templates/book_table.html:68 +msgid "Enter Series" +msgstr "Vnesi serijo" + +#: cps/templates/book_table.html:69 +msgid "Series Index" +msgstr "Številka serije" + +#: cps/templates/book_table.html:70 +msgid "Enter Languages" +msgstr "Vnesi jezike" + +#: cps/templates/book_table.html:71 +msgid "Publishing Date" +msgstr "Datum objave" + +#: cps/templates/book_table.html:72 +msgid "Enter Publishers" +msgstr "Vnesi založnike" + +#: cps/templates/book_table.html:73 +msgid "Enter comments" +msgstr "Vnesi komentarje" + +#: cps/templates/book_table.html:73 +msgid "Comments" +msgstr "Komentarji" + +#: cps/templates/book_table.html:75 +msgid "Archive Status" +msgstr "Stanje arhiva" + +#: cps/templates/book_table.html:77 cps/templates/search_form.html:42 +msgid "Read Status" +msgstr "Preberi stanje" + +#: cps/templates/book_table.html:80 cps/templates/book_table.html:82 +#: cps/templates/book_table.html:84 cps/templates/book_table.html:86 +#: cps/templates/book_table.html:90 cps/templates/book_table.html:92 +#: cps/templates/book_table.html:96 +msgid "Enter " +msgstr "Vnesi " + +#: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 +#: cps/templates/tasks.html:37 +msgid "Are you really sure?" +msgstr "Ste res prepričani?" + +#: cps/templates/book_table.html:117 +msgid "Books with Title will be merged from:" +msgstr "Knjige z naslovom bodo združene iz:" + +#: cps/templates/book_table.html:121 +msgid "Into Book with Title:" +msgstr "V knjigo z naslovom:" + +#: cps/templates/book_table.html:126 +msgid "Merge" +msgstr "Združitev" + +#: cps/templates/config_db.html:12 +msgid "Location of Calibre Database" +msgstr "Lokacija podatkovne zbirke Calibre" + +#: cps/templates/config_db.html:21 +msgid "Separate Book Files from Library" +msgstr "Loči knjižne datoteke od knjižnice" + +#: cps/templates/config_db.html:34 +msgid "Use Google Drive?" +msgstr "Uporabljaš Google Drive?" + +#: cps/templates/config_db.html:39 +msgid "Authenticate Google Drive" +msgstr "Preverjanje pristnosti storitve Google Drive" + +#: cps/templates/config_db.html:44 +msgid "Google Drive Calibre folder" +msgstr "Mapa Calibre v Google Drive" + +#: cps/templates/config_db.html:52 +msgid "Metadata Watch Channel ID" +msgstr "Metapodatki Watch ID kanala" + +#: cps/templates/config_db.html:55 +msgid "Revoke" +msgstr "Preklic" + +#: cps/templates/config_db.html:80 +msgid "New db location is invalid, please enter valid path" +msgstr "Nova lokacija db je neveljavna, vnesite veljavno pot" + +#: cps/templates/config_edit.html:18 +msgid "Server Configuration" +msgstr "Nastavitve strežnika" + +#: cps/templates/config_edit.html:25 +msgid "Server Port" +msgstr "Vrata strežnika" + +#: cps/templates/config_edit.html:28 +msgid "SSL certfile location (leave it empty for non-SSL Servers)" +msgstr "Lokacija datoteke SSL potrdila (za strežnike, ki ne uporabljajo SSL pustite prazno)" + +#: cps/templates/config_edit.html:35 +msgid "SSL Keyfile location (leave it empty for non-SSL Servers)" +msgstr "Lokacija datoteke s ključi SSL (za strežnike, ki ne uporabljajo SSL pustite prazno)" + +#: cps/templates/config_edit.html:43 +msgid "Update Channel" +msgstr "Kanal za posodobitve" + +#: cps/templates/config_edit.html:45 +msgid "Stable" +msgstr "Stabilen" + +#: cps/templates/config_edit.html:46 +msgid "Nightly" +msgstr "Nočen" + +#: cps/templates/config_edit.html:50 +msgid "Trusted Hosts (Comma Separated)" +msgstr "Zaupanja vredni gostitelji (ločeni z vejico)" + +#: cps/templates/config_edit.html:61 +msgid "Logfile Configuration" +msgstr "Nastavitev dnevniške datoteke" + +#: cps/templates/config_edit.html:77 +msgid "Location and name of logfile (calibre-web.log for no entry)" +msgstr "Lokacija in ime dnevniške datoteke (calibre-web.log, če ni vneseno)" + +#: cps/templates/config_edit.html:82 +msgid "Enable Access Log" +msgstr "Omogoči dnevnik dostopa" + +#: cps/templates/config_edit.html:85 +msgid "Location and name of access logfile (access.log for no entry)" +msgstr "Lokacija in ime dnevniške datoteke dostopa (access.log, če ni vneseno)" + +#: cps/templates/config_edit.html:96 +msgid "Feature Configuration" +msgstr "Nastavitve funkcij" + +#: cps/templates/config_edit.html:104 +msgid "Convert non-English characters in title and author while saving to disk" +msgstr "Pretvarjanje neangleških znakov v naslovu in avtorju med shranjevanjem na disk" + +#: cps/templates/config_edit.html:108 +msgid "Embed Metadata to Ebook File on Download/Conversion/e-mail (needs Calibre/Kepubify binaries)" +msgstr "Vstavljanje metapodatkov v datoteko e-knjige ob prenosu/pretvorbi/elektronski pošti (potrebujete binarne datoteke Calibre/Kepubify)" + +#: cps/templates/config_edit.html:112 +msgid "Enable Uploads" +msgstr "Omogoči nalaganje" + +#: cps/templates/config_edit.html:112 +msgid "(Please ensure that users also have upload permissions)" +msgstr "(prepričaj se, da imajo uporabniki tudi dovoljenja za nalaganje)" + +#: cps/templates/config_edit.html:116 +msgid "Allowed Upload Fileformats" +msgstr "Dovoljene vrste datotek za nalaganje" + +#: cps/templates/config_edit.html:122 +msgid "Enable Anonymous Browsing" +msgstr "Omogočanje anonimnega brskanja" + +#: cps/templates/config_edit.html:126 +msgid "Enable Public Registration" +msgstr "Omogočanje javne registracije" + +#: cps/templates/config_edit.html:131 +msgid "Use Email as Username" +msgstr "Uporabi e-pošto kot uporabniško ime" + +#: cps/templates/config_edit.html:136 +msgid "Enable Magic Link Remote Login" +msgstr "Omogoči oddaljeno prijavo prek povezave Magic Link" + +#: cps/templates/config_edit.html:141 +msgid "Enable Kobo sync" +msgstr "Omogoči sinhronizacijo s Kobo" + +#: cps/templates/config_edit.html:146 +msgid "Proxy unknown requests to Kobo Store" +msgstr "Posredovanje neznanih zahtev do trgovine Kobo" + +#: cps/templates/config_edit.html:149 +msgid "Server External Port (for port forwarded API calls)" +msgstr "Zunanja vrata strežnika (za posredovane klice API)" + +#: cps/templates/config_edit.html:157 +msgid "Use Goodreads" +msgstr "Uporabi Goodreads" + +#: cps/templates/config_edit.html:161 +msgid "Goodreads API Key" +msgstr "Ključ API Goodreads" + +#: cps/templates/config_edit.html:168 +msgid "Allow Reverse Proxy Authentication" +msgstr "Omogočanje avtentikacije povratnega posrednika" + +#: cps/templates/config_edit.html:179 +msgid "Login type" +msgstr "Vrsta prijave" + +#: cps/templates/config_edit.html:181 +msgid "Use Standard Authentication" +msgstr "Uporaba standardnega preverjanja pristnosti" + +#: cps/templates/config_edit.html:183 +msgid "Use LDAP Authentication" +msgstr "Uporaba overjanja LDAP" + +#: cps/templates/config_edit.html:186 +msgid "Use OAuth" +msgstr "Uporaba protokola OAuth" + +#: cps/templates/config_edit.html:193 +msgid "LDAP Server Host Name or IP Address" +msgstr "Gostiteljsko ime ali naslov IP strežnika LDAP" + +#: cps/templates/config_edit.html:197 +msgid "LDAP Server Port" +msgstr "Vrata strežnika LDAP" + +#: cps/templates/config_edit.html:201 +msgid "LDAP Encryption" +msgstr "Šifriranje LDAP" + +#: cps/templates/config_edit.html:204 +msgid "TLS" +msgstr "TLS" + +#: cps/templates/config_edit.html:205 +msgid "SSL" +msgstr "SSL" + +#: cps/templates/config_edit.html:209 +msgid "LDAP CACertificate Path (Only needed for Client Certificate Authentication)" +msgstr "Pot do CACcertifikata LDAP (potrebna samo za preverjanje pristnosti odjemalčevega certifikata)" + +#: cps/templates/config_edit.html:216 +msgid "LDAP Certificate Path (Only needed for Client Certificate Authentication)" +msgstr "Pot potrdila LDAP (potrebna samo za preverjanje pristnosti potrdila odjemalca)" + +#: cps/templates/config_edit.html:223 +msgid "LDAP Keyfile Path (Only needed for Client Certificate Authentication)" +msgstr "Pot do datoteke ključa LDAP (potrebna samo za preverjanje pristnosti odjemalčevega potrdila)" + +#: cps/templates/config_edit.html:232 +msgid "LDAP Authentication" +msgstr "Preverjanje pristnosti LDAP" + +#: cps/templates/config_edit.html:234 +msgid "Anonymous" +msgstr "Anonimno" + +#: cps/templates/config_edit.html:235 +msgid "Unauthenticated" +msgstr "Neavtentificirano" + +#: cps/templates/config_edit.html:236 +msgid "Simple" +msgstr "Enostavno" + +#: cps/templates/config_edit.html:241 +msgid "LDAP Administrator Username" +msgstr "Uporabniško ime skrbnika LDAP" + +#: cps/templates/config_edit.html:247 +msgid "LDAP Administrator Password" +msgstr "Geslo skrbnika LDAP" + +#: cps/templates/config_edit.html:252 +msgid "LDAP Distinguished Name (DN)" +msgstr "Razpoznavno ime (DN) LDAP" + +#: cps/templates/config_edit.html:256 +msgid "LDAP User Object Filter" +msgstr "Filtriranje predmeta uporabnika LDAP" + +#: cps/templates/config_edit.html:261 +msgid "LDAP Server is OpenLDAP?" +msgstr "Strežnik LDAP je OpenLDAP?" + +#: cps/templates/config_edit.html:263 +msgid "Following Settings are Needed For User Import" +msgstr "Za uvoz uporabnika so potrebne naslednje nastavitve" + +#: cps/templates/config_edit.html:265 +msgid "LDAP Group Object Filter" +msgstr "Filter objektov skupine LDAP" + +#: cps/templates/config_edit.html:269 +msgid "LDAP Group Name" +msgstr "Ime skupine LDAP" + +#: cps/templates/config_edit.html:273 +msgid "LDAP Group Members Field" +msgstr "Polje člani skupine LDAP" + +#: cps/templates/config_edit.html:277 +msgid "LDAP Member User Filter Detection" +msgstr "Zaznavanje filtrov za uporabnike članov LDAP" + +#: cps/templates/config_edit.html:279 +msgid "Autodetect" +msgstr "Samodejno zaznavanje" + +#: cps/templates/config_edit.html:280 +msgid "Custom Filter" +msgstr "Filter po meri" + +#: cps/templates/config_edit.html:285 +msgid "LDAP Member User Filter" +msgstr "Filtriranje uporabnikov članov LDAP" + +#: cps/templates/config_edit.html:296 +#, python-format +msgid "Obtain %(provider)s OAuth Credential" +msgstr "Pridobitev %(provider)s poverilnice OAuth" + +#: cps/templates/config_edit.html:299 +#, python-format +msgid "%(provider)s OAuth Client Id" +msgstr "%(provider)s Id odjemalca OAuth" + +#: cps/templates/config_edit.html:303 +#, python-format +msgid "%(provider)s OAuth Client Secret" +msgstr "%(provider)s skrivnost odjemalca OAuth" + +#: cps/templates/config_edit.html:319 +msgid "External binaries" +msgstr "Zunanje binarne datoteke" + +#: cps/templates/config_edit.html:325 +msgid "Path to Calibre Binaries" +msgstr "Pot do binarnih datotek Calibre" + +#: cps/templates/config_edit.html:333 +msgid "Calibre E-Book Converter Settings" +msgstr "Nastavitve pretvornika e-knjig Calibre" + +#: cps/templates/config_edit.html:336 +msgid "Path to Kepubify E-Book Converter" +msgstr "Pot do Kepubify pretvornika e-knjig" + +#: cps/templates/config_edit.html:344 +msgid "Location of Unrar binary" +msgstr "Lokacija binarnega programa Unrar" + +#: cps/templates/config_edit.html:360 +msgid "Security Settings" +msgstr "Varnostne nastavitve" + +#: cps/templates/config_edit.html:368 +msgid "Limit failed login attempts" +msgstr "Omejitev neuspelih poskusov prijave" + +#: cps/templates/config_edit.html:372 +msgid "Configure Backend for Limiter" +msgstr "Nastavitev zalednega sistema za omejitve" + +#: cps/templates/config_edit.html:376 +msgid "Options for Limiter Backend" +msgstr "Možnosti za omejitev zalednega sistema" + +#: cps/templates/config_edit.html:382 +msgid "Check if file extensions matches file content on upload" +msgstr "Preveri ali se končnice datotek ujemajo z vsebino datoteke pri nalaganju" + +#: cps/templates/config_edit.html:385 +msgid "Session protection" +msgstr "Zaščita seje" + +#: cps/templates/config_edit.html:387 +msgid "Basic" +msgstr "Osnovna" + +#: cps/templates/config_edit.html:388 +msgid "Strong" +msgstr "Močna" + +#: cps/templates/config_edit.html:393 +msgid "User Password policy" +msgstr "Pravilnik o geslih uporabnikov" + +#: cps/templates/config_edit.html:397 +msgid "Minimum password length" +msgstr "Najmanjša dolžina gesla" + +#: cps/templates/config_edit.html:402 +msgid "Enforce number" +msgstr "Obvezna uporaba številk" + +#: cps/templates/config_edit.html:406 +msgid "Enforce lowercase characters" +msgstr "Obvezna uporaba malih črk" + +#: cps/templates/config_edit.html:410 +msgid "Enforce uppercase characters" +msgstr "Obvezna uporaba velikih črk" + +#: cps/templates/config_edit.html:414 +msgid "Enforce characters (needed For Chinese/Japanese/Korean Characters)" +msgstr "Obvezna uporaba znakov (potrebno za kitajske/japonske/korejske znake)" + +#: cps/templates/config_edit.html:418 +msgid "Enforce special characters" +msgstr "Obvezna uporaba posebnih znakov" + +#: cps/templates/config_view_edit.html:17 +msgid "View Configuration" +msgstr "Prikaži nastavitve" + +#: cps/templates/config_view_edit.html:32 +msgid "No. of Random Books to Display" +msgstr "Število naključnih knjig za prikaz" + +#: cps/templates/config_view_edit.html:36 +msgid "No. of Authors to Display Before Hiding (0=Disable Hiding)" +msgstr "Število avtorjev, ki se prikažejo pred skrivanjem (0 = onemogoči skrivanje)" + +#: cps/templates/config_view_edit.html:40 cps/templates/readcbr.html:101 +msgid "Theme" +msgstr "Tema" + +#: cps/templates/config_view_edit.html:42 +msgid "Standard Theme" +msgstr "Standardna tema" + +#: cps/templates/config_view_edit.html:43 +msgid "caliBlur! Dark Theme" +msgstr "caliBlur! temna tema" + +#: cps/templates/config_view_edit.html:47 +msgid "Regular Expression for Ignoring Columns" +msgstr "Regularni izraz za ignoriranje stolpcev" + +#: cps/templates/config_view_edit.html:51 +msgid "Link Read/Unread Status to Calibre Column" +msgstr "Poveži stanje prebrano/neprebrano s stolpcem Calibre" + +#: cps/templates/config_view_edit.html:60 +msgid "View Restrictions based on Calibre column" +msgstr "Prikaz omejitev na podlagi stolpca Calibre" + +#: cps/templates/config_view_edit.html:69 +msgid "Regular Expression for Title Sorting" +msgstr "Regularni izraz za razvrščanje naslovov" + +#: cps/templates/config_view_edit.html:80 +msgid "Default Settings for New Users" +msgstr "Privzete nastavitve za nove uporabnike" + +#: cps/templates/config_view_edit.html:88 cps/templates/user_edit.html:96 +msgid "Admin User" +msgstr "Administratorski uporabnik" + +#: cps/templates/config_view_edit.html:92 cps/templates/user_edit.html:101 +msgid "Allow Downloads" +msgstr "Dovoli prenose" + +#: cps/templates/config_view_edit.html:96 cps/templates/user_edit.html:105 +msgid "Allow eBook Viewer" +msgstr "Dovoli pregledovalnik e-knjig" + +#: cps/templates/config_view_edit.html:101 cps/templates/user_edit.html:110 +msgid "Allow Uploads" +msgstr "Dovoli nalaganje" + +#: cps/templates/config_view_edit.html:106 cps/templates/user_edit.html:115 +msgid "Allow Edit" +msgstr "Omogoči urejanje" + +#: cps/templates/config_view_edit.html:111 cps/templates/user_edit.html:120 +msgid "Allow Delete Books" +msgstr "Dovoli izbris knjig" + +#: cps/templates/config_view_edit.html:116 cps/templates/user_edit.html:126 +msgid "Allow Changing Password" +msgstr "Dovoli spreminjanje gesla" + +#: cps/templates/config_view_edit.html:120 cps/templates/user_edit.html:130 +msgid "Allow Editing Public Shelves" +msgstr "Omogoči urejanje javnih polic" + +#: cps/templates/config_view_edit.html:123 +msgid "Default Language" +msgstr "Privzeti jezik" + +#: cps/templates/config_view_edit.html:131 +msgid "Default Visible Language of Books" +msgstr "Privzet vidni jezik knjig" + +#: cps/templates/config_view_edit.html:147 +msgid "Default Visibilities for New Users" +msgstr "Privzete vidnosti za nove uporabnike" + +#: cps/templates/config_view_edit.html:163 cps/templates/user_edit.html:84 +#: cps/templates/user_table.html:154 +msgid "Show Random Books in Detail View" +msgstr "Prikaži naključne knjige v podrobnem pogledu" + +#: cps/templates/config_view_edit.html:166 cps/templates/user_edit.html:87 +msgid "Add Allowed/Denied Tags" +msgstr "Dodajanje oznak dovoljeno/zavrnjeno" + +#: cps/templates/config_view_edit.html:167 +msgid "Add Allowed/Denied custom column values" +msgstr "Dodajanje vrednosti stolpca dovoljeno/zavrnjeno po meri" + +#: cps/templates/detail.html:85 cps/templates/detail.html:99 +msgid "Read in Browser" +msgstr "Preberi v brskalniku" + +#: cps/templates/detail.html:108 cps/templates/detail.html:128 +msgid "Listen in Browser" +msgstr "Poslušaj v brskalniku" + +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 +msgid "Mark As Unread" +msgstr "Označi kot neprebrano" + +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 +msgid "Mark As Read" +msgstr "Označi kot prebrano" + +#: cps/templates/detail.html:262 +msgid "Mark Book as Read or Unread" +msgstr "Označite knjigo kot prebrano ali neprebrano" + +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 +msgid "Read" +msgstr "Preberi" + +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 +msgid "Restore from archive" +msgstr "Obnovi iz arhiva" + +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 +msgid "Add to archive" +msgstr "Dodaj v arhiv" + +#: cps/templates/detail.html:275 +msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" +msgstr "Označi knjigo kot arhivirano ali ne, da se skrije v Calibre-Web in izbriše iz Kobo bralnika" + +#: cps/templates/detail.html:275 +msgid "Archive" +msgstr "Arhiv" + +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 +#: cps/templates/search.html:16 +msgid "Add to shelf" +msgstr "Dodaj na polico" + +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 +#: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 +#: cps/templates/search.html:22 +msgid "(Public)" +msgstr "(Javno)" + +#: cps/templates/detail.html:348 +msgid "Edit Metadata" +msgstr "Urejanje metapodatkov" + +#: cps/templates/email_edit.html:13 +msgid "Email Account Type" +msgstr "Vrsta e-poštnega računa" + +#: cps/templates/email_edit.html:15 +msgid "Standard Email Account" +msgstr "Standardni e-poštni račun" + +#: cps/templates/email_edit.html:16 +msgid "Gmail Account" +msgstr "Račun Gmail" + +#: cps/templates/email_edit.html:22 +msgid "Setup Gmail Account" +msgstr "Nastavitev računa Gmail" + +#: cps/templates/email_edit.html:24 +msgid "Revoke Gmail Access" +msgstr "Preklic dostopov do Gmail" + +#: cps/templates/email_edit.html:42 +msgid "STARTTLS" +msgstr "STARTTLS" + +#: cps/templates/email_edit.html:43 +msgid "SSL/TLS" +msgstr "SSL/TLS" + +#: cps/templates/email_edit.html:51 +msgid "SMTP Password" +msgstr "Geslo za SMTP" + +#: cps/templates/email_edit.html:58 +msgid "Attachment Size Limit" +msgstr "Omejitev velikosti priloge" + +#: cps/templates/email_edit.html:66 +msgid "Save and Send Test Email" +msgstr "Shrani in pošlji testno e-pošto" + +#: cps/templates/email_edit.html:70 cps/templates/layout.html:26 +#: cps/templates/shelf_order.html:42 cps/templates/user_table.html:174 +msgid "Back" +msgstr "Nazaj" + +#: cps/templates/email_edit.html:74 +msgid "Allowed Domains (Whitelist)" +msgstr "Dovoljene domene (bel seznam)" + +#: cps/templates/email_edit.html:78 cps/templates/email_edit.html:105 +msgid "Add Domain" +msgstr "Dodajanje domene" + +#: cps/templates/email_edit.html:81 cps/templates/email_edit.html:108 +#: cps/templates/user_table.html:27 +msgid "Add" +msgstr "Dodaj" + +#: cps/templates/email_edit.html:86 cps/templates/email_edit.html:96 +msgid "Enter domainname" +msgstr "Vnesi ime domene" + +#: cps/templates/email_edit.html:92 +msgid "Denied Domains (Blacklist)" +msgstr "Zavrnjene domene (črni seznam)" + +#: cps/templates/generate_kobo_auth_url.html:6 +msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" +msgstr "V urejevalniku besedila odprite datoteko .kobo/Kobo/Kobo eReader.conf in dodajte (ali uredite):" + +#: cps/templates/generate_kobo_auth_url.html:11 +msgid "Kobo Token:" +msgstr "Kobo žeton:" + +#: cps/templates/grid.html:21 +msgid "List" +msgstr "Seznam" + +#: cps/templates/http_error.html:34 +msgid "Calibre-Web Instance is unconfigured, please contact your administrator" +msgstr "Instanca Calibre-Web ni nastavljena, obrni se na skrbnika" + +#: cps/templates/http_error.html:44 +msgid "Create Issue" +msgstr "Ustvarjanje poročila o težavi" + +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Nastavitev zbirke podatkov" + +#: cps/templates/http_error.html:54 +msgid "Return to Home" +msgstr "Vrnitev domov" + +#: cps/templates/http_error.html:57 +msgid "Logout User" +msgstr "Odjava uporabnika" + +#: cps/templates/index.html:71 +msgid "Sort ascending according to download count" +msgstr "Razvrsti naraščajoče glede na število prenosov" + +#: cps/templates/index.html:72 +msgid "Sort descending according to download count" +msgstr "Razvrsti padajoče glede na število prenosov" + +#: cps/templates/index.html:78 cps/templates/search.html:35 +#: cps/templates/shelf.html:24 +msgid "Sort authors in alphabetical order" +msgstr "Razvrsti avtorje po abecednem vrstnem redu" + +#: cps/templates/index.html:79 cps/templates/search.html:36 +#: cps/templates/shelf.html:25 +msgid "Sort authors in reverse alphabetical order" +msgstr "Razvrsti avtorje v obratnem abecednem vrstnem redu" + +#: cps/templates/index.html:83 +msgid "Sort ascending according to series index" +msgstr "Razvrsti naraščajoče glede na indeks serije" + +#: cps/templates/index.html:84 +msgid "Sort descending according to series index" +msgstr "Razvrsti padajoče glede na indeks serije" + +#: cps/templates/index.xml:7 +msgid "Start" +msgstr "Začetek" + +#: cps/templates/index.xml:19 +msgid "Alphabetical Books" +msgstr "Abecedno razvrščene knjige" + +#: cps/templates/index.xml:23 +msgid "Books sorted alphabetically" +msgstr "Knjige razvrščene po abecedi" + +#: cps/templates/index.xml:31 +msgid "Popular publications from this catalog based on Downloads." +msgstr "Priljubljene publikacije iz tega kataloga na podlagi prenosov." + +#: cps/templates/index.xml:40 +msgid "Popular publications from this catalog based on Rating." +msgstr "Priljubljene publikacije iz tega kataloga na podlagi ocene." + +#: cps/templates/index.xml:45 +msgid "Recently added Books" +msgstr "Nedavno dodane knjige" + +#: cps/templates/index.xml:49 +msgid "The latest Books" +msgstr "Najnovejše knjige" + +#: cps/templates/index.xml:54 +msgid "Random Books" +msgstr "Naključne knjige" + +#: cps/templates/index.xml:83 +msgid "Books ordered by Author" +msgstr "Knjige, razvrščene po avtorju" + +#: cps/templates/index.xml:92 +msgid "Books ordered by publisher" +msgstr "Knjige, razvrščene po založniku" + +#: cps/templates/index.xml:101 +msgid "Books ordered by category" +msgstr "Knjige, razvrščene po kategorijah" + +#: cps/templates/index.xml:110 +msgid "Books ordered by series" +msgstr "Knjige, razvrščene po serijah" + +#: cps/templates/index.xml:119 +msgid "Books ordered by Languages" +msgstr "Knjige, razvrščene po jezikih" + +#: cps/templates/index.xml:128 +msgid "Books ordered by Rating" +msgstr "Knjige, razvrščene po oceni" + +#: cps/templates/index.xml:137 +msgid "Books ordered by file formats" +msgstr "Knjige, razvrščene po vrstah datotek" + +#: cps/templates/index.xml:142 cps/templates/layout.html:155 +#: cps/templates/search_form.html:88 +msgid "Shelves" +msgstr "Police" + +#: cps/templates/index.xml:146 +msgid "Books organized in shelves" +msgstr "Knjige, urejene po policah" + +#: cps/templates/layout.html:32 +msgid "Toggle Navigation" +msgstr "Preklopi navigacijo" + +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Enostavno" + +#: cps/templates/layout.html:67 cps/templates/layout.html:97 +msgid "Account" +msgstr "Račun" + +#: cps/templates/layout.html:94 cps/templates/read.html:78 +#: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 +msgid "Settings" +msgstr "Nastavitve" + +#: cps/templates/layout.html:138 +msgid "Please do not refresh the page" +msgstr "Ne osvežuj strani" + +#: cps/templates/layout.html:148 +msgid "Browse" +msgstr "Brskaj" + +#: cps/templates/layout.html:161 cps/templates/stats.html:3 +msgid "About" +msgstr "O programu" + +#: cps/templates/layout.html:202 +msgid "Book Details" +msgstr "Podrobnosti o knjigi" + +#: cps/templates/list.html:22 +msgid "Grid" +msgstr "Mreža" + +#: cps/templates/listenmp3.html:167 +msgid "Archived" +msgstr "Arhivirano" + +#: cps/templates/login.html:18 +msgid "Remember Me" +msgstr "Zapomni si me" + +#: cps/templates/login.html:23 +msgid "Forgot Password?" +msgstr "Ste pozabili geslo?" + +#: cps/templates/login.html:34 +msgid "Log in with Magic Link" +msgstr "Prijavi se s povezavo Magic Link" + +#: cps/templates/logviewer.html:6 +msgid "Show Calibre-Web Log: " +msgstr "Prikaži dnevnik Calibre-Web:" + +#: cps/templates/logviewer.html:8 +msgid "Calibre-Web Log: " +msgstr "Dnevnik Calibre-Web:" + +#: cps/templates/logviewer.html:8 +msgid "Stream output, can't be displayed" +msgstr "Izhodni tok, ki ga ni mogoče prikazati" + +#: cps/templates/logviewer.html:12 +msgid "Show Access Log: " +msgstr "Prikaži dnevnik dostopa: " + +#: cps/templates/logviewer.html:18 +msgid "Download Calibre-Web Log" +msgstr "Prenos Calibre-Web Log" + +#: cps/templates/logviewer.html:21 +msgid "Download Access Log" +msgstr "Prenos dnevnika dostopa" + +#: cps/templates/modal_dialogs.html:6 +msgid "Select Allowed/Denied Tags" +msgstr "Izberite dovoljene/zavrnjene oznake" + +#: cps/templates/modal_dialogs.html:7 +msgid "Select Allowed/Denied Custom Column Values" +msgstr "Izberite dovoljene/zavrnjene vrednosti stolpcev po meri" + +#: cps/templates/modal_dialogs.html:8 +msgid "Select Allowed/Denied Tags of User" +msgstr "Izberite dovoljene/zavrnjene oznake uporabnika" + +#: cps/templates/modal_dialogs.html:9 +msgid "Select Allowed/Denied Custom Column Values of User" +msgstr "Izberite dovoljene/zavrnjene vrednosti stolpcev po meri uporabnika" + +#: cps/templates/modal_dialogs.html:15 +msgid "Enter Tag" +msgstr "Vnesite oznako" + +#: cps/templates/modal_dialogs.html:24 +msgid "Add View Restriction" +msgstr "Dodajanje omejitve prikaza" + +#: cps/templates/modal_dialogs.html:50 +msgid "This book format will be permanently erased from database" +msgstr "Ta vrsta knjige bo trajno izbrisana iz podatkovne baze" + +#: cps/templates/modal_dialogs.html:51 +msgid "This book will be permanently erased from database" +msgstr "Ta knjiga bo trajno izbrisana iz baze podatkov" + +#: cps/templates/modal_dialogs.html:52 +msgid "and hard disk" +msgstr "in iz trdega diska" + +#: cps/templates/modal_dialogs.html:56 +msgid "Important Kobo Note: deleted books will remain on any paired Kobo device." +msgstr "Pomembno opozorilo Kobo: izbrisane knjige bodo ostale v vsaki povezani napravi Kobo." + +#: cps/templates/modal_dialogs.html:57 +msgid "Books must first be archived and the device synced before a book can safely be deleted." +msgstr "Preden lahko knjigo varno izbrišete, jo morate najprej arhivirati in sinhronizirati napravo." + +#: cps/templates/modal_dialogs.html:76 +msgid "Choose File Location" +msgstr "Izberi lokacijo datoteke" + +#: cps/templates/modal_dialogs.html:82 +msgid "type" +msgstr "vrsta" + +#: cps/templates/modal_dialogs.html:83 +msgid "name" +msgstr "ime" + +#: cps/templates/modal_dialogs.html:84 +msgid "size" +msgstr "velikost" + +#: cps/templates/modal_dialogs.html:90 +msgid "Parent Directory" +msgstr "Nadrejeni imenik" + +#: cps/templates/modal_dialogs.html:98 +msgid "Select" +msgstr "Izberi" + +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 +msgid "Ok" +msgstr "V redu" + +#: cps/templates/osd.xml:5 +msgid "Calibre-Web eBook Catalog" +msgstr "Katalog e-knjig Calibre-Web" + +#: cps/templates/read.html:7 +msgid "epub Reader" +msgstr "epub bralnik" + +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Izberite uporabniško ime" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 +msgid "Light" +msgstr "Svetlo" + +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 +msgid "Dark" +msgstr "Temno" + +#: cps/templates/read.html:88 +msgid "Sepia" +msgstr "Sepia" + +#: cps/templates/read.html:90 +msgid "Black" +msgstr "Črno" + +#: cps/templates/read.html:95 +msgid "Reflow text when sidebars are open." +msgstr "Preusmerjanje besedila, ko so odprte stranske vrstice." + +#: cps/templates/read.html:100 +msgid "Font Sizes" +msgstr "Velikosti pisave" + +#: cps/templates/read.html:105 +msgid "Font" +msgstr "Pisava" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "Privzeto" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "Yahei" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "SimSun" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "KaiTi" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "Razprši" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "Dva stolpca" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "En stolpec" + +#: cps/templates/readcbr.html:8 +msgid "Comic Reader" +msgstr "Bralnik stripov" + +#: cps/templates/readcbr.html:75 +msgid "Keyboard Shortcuts" +msgstr "Bližnjice na tipkovnici" + +#: cps/templates/readcbr.html:78 +msgid "Previous Page" +msgstr "Prejšnja stran" + +#: cps/templates/readcbr.html:79 cps/templates/readcbr.html:159 +msgid "Next Page" +msgstr "Naslednja stran" + +#: cps/templates/readcbr.html:80 +msgid "Single Page Display" +msgstr "Prikaz ene strani" + +#: cps/templates/readcbr.html:81 +msgid "Long Strip Display" +msgstr "Prikaz dolgega traku" + +#: cps/templates/readcbr.html:82 +msgid "Scale to Best" +msgstr "Povečaj na najboljše" + +#: cps/templates/readcbr.html:83 +msgid "Scale to Width" +msgstr "Povečaj na širino" + +#: cps/templates/readcbr.html:84 +msgid "Scale to Height" +msgstr "Povečaj na višino" + +#: cps/templates/readcbr.html:85 +msgid "Scale to Native" +msgstr "Povečaj na privzeto" + +#: cps/templates/readcbr.html:86 +msgid "Rotate Right" +msgstr "Obračanje v desno" + +#: cps/templates/readcbr.html:87 +msgid "Rotate Left" +msgstr "Obračanje v levo" + +#: cps/templates/readcbr.html:88 +msgid "Flip Image" +msgstr "Obrni sliko" + +#: cps/templates/readcbr.html:110 +msgid "Display" +msgstr "Prikaz" + +#: cps/templates/readcbr.html:113 +msgid "Single Page" +msgstr "Ena stran" + +#: cps/templates/readcbr.html:114 +msgid "Long Strip" +msgstr "Dolgi trak" + +#: cps/templates/readcbr.html:119 +msgid "Scale" +msgstr "Povečava" + +#: cps/templates/readcbr.html:122 +msgid "Best" +msgstr "Najboljša" + +#: cps/templates/readcbr.html:123 +msgid "Width" +msgstr "Širina" + +#: cps/templates/readcbr.html:124 +msgid "Height" +msgstr "Višina" + +#: cps/templates/readcbr.html:125 +msgid "Native" +msgstr "Privzeto" + +#: cps/templates/readcbr.html:130 +msgid "Rotate" +msgstr "Zavrti" + +#: cps/templates/readcbr.html:141 +msgid "Flip" +msgstr "Obrni" + +#: cps/templates/readcbr.html:144 +msgid "Horizontal" +msgstr "Vodoravno" + +#: cps/templates/readcbr.html:145 +msgid "Vertical" +msgstr "Navpično" + +#: cps/templates/readcbr.html:150 +msgid "Direction" +msgstr "Smer" + +#: cps/templates/readcbr.html:153 +msgid "Left to Right" +msgstr "Od leve proti desni" + +#: cps/templates/readcbr.html:154 +msgid "Right to Left" +msgstr "Od desne proti levi" + +#: cps/templates/readcbr.html:162 +msgid "Reset to Top" +msgstr "Ponastavitev na vrh" + +#: cps/templates/readcbr.html:163 +msgid "Remember Position" +msgstr "Zapomni si položaj" + +#: cps/templates/readcbr.html:168 +msgid "Scrollbar" +msgstr "Drsni trak" + +#: cps/templates/readcbr.html:171 +msgid "Show" +msgstr "Prikaži" + +#: cps/templates/readcbr.html:172 +msgid "Hide" +msgstr "Skrij" + +#: cps/templates/readdjvu.html:5 +msgid "DJVU Reader" +msgstr "Bralnik DJVU" + +#: cps/templates/readpdf.html:31 +msgid "PDF Reader" +msgstr "Bralnik PDF" + +#: cps/templates/readtxt.html:6 +msgid "txt Reader" +msgstr "txt bralnik" + +#: cps/templates/register.html:4 +msgid "Register New Account" +msgstr "Registracija novega računa" + +#: cps/templates/register.html:10 +msgid "Choose a username" +msgstr "Izberite uporabniško ime" + +#: cps/templates/register.html:15 +msgid "Your Email" +msgstr "Vaš e-poštni naslov" + +#: cps/templates/remote_login.html:5 +msgid "Magic Link - Authorise New Device" +msgstr "Magic Link - avtorizacija nove naprave" + +#: cps/templates/remote_login.html:7 +msgid "On another device, login and visit:" +msgstr "V drugi napravi se prijavite in obiščite:" + +#: cps/templates/remote_login.html:11 +msgid "Once verified, you will automatically be logged in on this device." +msgstr "Po preverjanju boste samodejno prijavljeni v to napravo." + +#: cps/templates/remote_login.html:14 +msgid "This verification link will expire in 10 minutes." +msgstr "Ta povezava za preverjanje poteče čez 10 minut." + +#: cps/templates/schedule_edit.html:33 +msgid "Generate Series Cover Thumbnails" +msgstr "Ustvarjanje sličic naslovnic serij" + +#: cps/templates/search.html:7 +msgid "Search Term:" +msgstr "Iskalni izraz:" + +#: cps/templates/search.html:9 +msgid "Results for:" +msgstr "Rezultati za:" + +#: cps/templates/search_form.html:21 +msgid "Published Date From" +msgstr "Datum objave od" + +#: cps/templates/search_form.html:31 +msgid "Published Date To" +msgstr "Datum objave do" + +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 +msgid "Any" +msgstr "Vse" + +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 +msgid "Empty" +msgstr "Prazen" + +#: cps/templates/search_form.html:60 +msgid "Exclude Tags" +msgstr "Izključi oznake" + +#: cps/templates/search_form.html:78 +msgid "Exclude Series" +msgstr "Izključi serijo" + +#: cps/templates/search_form.html:96 +msgid "Exclude Shelves" +msgstr "Izključi police" + +#: cps/templates/search_form.html:116 +msgid "Exclude Languages" +msgstr "Izključi jezike" + +#: cps/templates/search_form.html:127 +msgid "Extensions" +msgstr "Razširitve" + +#: cps/templates/search_form.html:135 +msgid "Exclude Extensions" +msgstr "Izključitev razširitev" + +#: cps/templates/search_form.html:145 +msgid "Rating Above" +msgstr "Zgornja ocena" + +#: cps/templates/search_form.html:149 +msgid "Rating Below" +msgstr "Spodnja ocena" + +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 +msgid "From:" +msgstr "Od:" + +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 +msgid "To:" +msgstr "Za:" + +#: cps/templates/shelf.html:13 +msgid "Delete this Shelf" +msgstr "Brisanje te police" + +#: cps/templates/shelf.html:14 +msgid "Edit Shelf Properties" +msgstr "Urejanje lastnosti police" + +#: cps/templates/shelf.html:17 +msgid "Arrange books manually" +msgstr "Ročno urejanje knjig" + +#: cps/templates/shelf.html:18 +msgid "Disable Change order" +msgstr "Onemogoči spremembo vrstnega reda" + +#: cps/templates/shelf.html:18 +msgid "Enable Change order" +msgstr "Omogoči spremembo vrstnega reda" + +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "Razvrsti glede na knjigo, ki je bila dodana na polico, najprej najnovejša." + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "Razvrsti glede na knjigo, ki je bila dodana na polico, najprej najstarejša." + +#: cps/templates/shelf_edit.html:14 +msgid "Share with Everyone" +msgstr "Deli z vsemi" + +#: cps/templates/shelf_edit.html:21 +msgid "Sync this shelf with Kobo device" +msgstr "Sinhronizacija te police z napravo Kobo" + +#: cps/templates/shelf_order.html:5 +msgid "Drag to Rearrange Order" +msgstr "Povlecite, da spremenite vrstni red" + +#: cps/templates/shelf_order.html:33 +msgid "Hidden Book" +msgstr "Skrita knjiga" + +#: cps/templates/stats.html:7 +msgid "Library Statistics" +msgstr "Statistika knjižnice" + +#: cps/templates/stats.html:12 +msgid "Books in this Library" +msgstr "Knjig v tej knjižnici" + +#: cps/templates/stats.html:16 +msgid "Authors in this Library" +msgstr "Avtorjev v tej knjižnici" + +#: cps/templates/stats.html:20 +msgid "Categories in this Library" +msgstr "Kategorij v tej knjižnici" + +#: cps/templates/stats.html:24 +msgid "Series in this Library" +msgstr "Serij v tej knjižnici" + +#: cps/templates/stats.html:29 +msgid "System Statistics" +msgstr "Sistemske statistike" + +#: cps/templates/stats.html:33 +msgid "Program" +msgstr "Program" + +#: cps/templates/stats.html:34 +msgid "Installed Version" +msgstr "Nameščena različica" + +#: cps/templates/tasks.html:12 +msgid "User" +msgstr "Uporabnik" + +#: cps/templates/tasks.html:14 +msgid "Task" +msgstr "Opravilo" + +#: cps/templates/tasks.html:15 +msgid "Status" +msgstr "Status" + +#: cps/templates/tasks.html:16 +msgid "Progress" +msgstr "Napredek" + +#: cps/templates/tasks.html:17 +msgid "Run Time" +msgstr "Čas delovanja" + +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "Sporočilo" + +#: cps/templates/tasks.html:21 +msgid "Actions" +msgstr "Dejanja" + +#: cps/templates/tasks.html:41 +msgid "This task will be cancelled. Any progress made by this task will be saved." +msgstr "To opravilo bo preklicano. Vsak napredek, dosežen pri tem opravilu, bo shranjen." + +#: cps/templates/tasks.html:42 +msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." +msgstr "Če je to načrtovano opravilo, se bo ponovno izvedlo ob naslednjem načrtovanem času." + +#: cps/templates/user_edit.html:20 +msgid "Reset user Password" +msgstr "Ponastavitev uporabniškega gesla" + +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "Pošlji v e-bralnikov e-naslov. Uporabi vejico za ločitev e-poštnih sporočil za več e-bralnikov" + +#: cps/templates/user_edit.html:43 +msgid "Language of Books" +msgstr "Jezik knjig" + +#: cps/templates/user_edit.html:54 +msgid "OAuth Settings" +msgstr "Nastavitve OAuth" + +#: cps/templates/user_edit.html:56 +msgid "Link" +msgstr "Povezava" + +#: cps/templates/user_edit.html:58 +msgid "Unlink" +msgstr "Odklop povezave" + +#: cps/templates/user_edit.html:64 +msgid "Kobo Sync Token" +msgstr "Žeton za sinhronizacijo Kobo" + +#: cps/templates/user_edit.html:66 +msgid "Create/View" +msgstr "Ustvari/pogled" + +#: cps/templates/user_edit.html:70 +msgid "Force full kobo sync" +msgstr "Vsili popolno sinhronizacijo s Kobo" + +#: cps/templates/user_edit.html:88 +msgid "Add allowed/Denied Custom Column Values" +msgstr "Dodajanje dovoljenih/zavrnjenih vrednosti stolpcev po meri" + +#: cps/templates/user_edit.html:137 +msgid "Sync only books in selected shelves with Kobo" +msgstr "Sinhroniziranje samo knjig na izbranih policah s Kobo" + +#: cps/templates/user_edit.html:147 cps/templates/user_table.html:169 +msgid "Delete User" +msgstr "Izbriši uporabnika" + +#: cps/templates/user_edit.html:159 +msgid "Generate Kobo Auth URL" +msgstr "Ustvarite URL avtentikacije Kobo" + +#: cps/templates/user_table.html:80 cps/templates/user_table.html:103 +msgid "Select..." +msgstr "Izberi..." + +#: cps/templates/user_table.html:131 +msgid "Edit User" +msgstr "Uredi uporabnika" + +#: cps/templates/user_table.html:134 +msgid "Enter Username" +msgstr "Vnesi uporabniško ime" + +#: cps/templates/user_table.html:135 +msgid "Enter Email" +msgstr "Vnesi e-pošto" + +#: cps/templates/user_table.html:136 +msgid "Enter eReader Email" +msgstr "Vnesi e-pošto e-bralnika" + +#: cps/templates/user_table.html:136 +msgid "eReader Email" +msgstr "E-pošta za e-bralnik" + +#: cps/templates/user_table.html:137 +msgid "Locale" +msgstr "Lokacija" + +#: cps/templates/user_table.html:138 +msgid "Visible Book Languages" +msgstr "Jeziki vidnih knjig" + +#: cps/templates/user_table.html:139 +msgid "Edit Allowed Tags" +msgstr "Uredi dovoljene oznake" + +#: cps/templates/user_table.html:139 +msgid "Allowed Tags" +msgstr "Dovoljene oznake" + +#: cps/templates/user_table.html:140 +msgid "Edit Denied Tags" +msgstr "Uredi zavrnjene oznake" + +#: cps/templates/user_table.html:140 +msgid "Denied Tags" +msgstr "Zavrnjene oznake" + +#: cps/templates/user_table.html:141 +msgid "Edit Allowed Column Values" +msgstr "Urejanje dovoljenih vrednosti stolpcev" + +#: cps/templates/user_table.html:141 +msgid "Allowed Column Values" +msgstr "Dovoljene vrednosti stolpcev" + +#: cps/templates/user_table.html:142 +msgid "Edit Denied Column Values" +msgstr "Urejanje vrednosti zavrnjenega stolpca" + +#: cps/templates/user_table.html:142 +msgid "Denied Column Values" +msgstr "Zavrnjene vrednosti stolpcev" + +#: cps/templates/user_table.html:144 +msgid "Change Password" +msgstr "Sprememba gesla" + +#: cps/templates/user_table.html:147 +msgid "View" +msgstr "Oglejte si" + +#: cps/templates/user_table.html:150 +msgid "Edit Public Shelves" +msgstr "Urejanje javnih polic" + +#: cps/templates/user_table.html:152 +msgid "Sync selected Shelves with Kobo" +msgstr "Usklajevanje izbrane police s Kobo" + +#: cps/templates/user_table.html:156 +msgid "Show Read/Unread Section" +msgstr "Prikaži razdelek Prebrano/neprebrano" + diff --git a/cps/translations/sv/LC_MESSAGES/messages.mo b/cps/translations/sv/LC_MESSAGES/messages.mo index fc794289..ee7e9097 100644 Binary files a/cps/translations/sv/LC_MESSAGES/messages.mo and b/cps/translations/sv/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/sv/LC_MESSAGES/messages.po b/cps/translations/sv/LC_MESSAGES/messages.po index 885adce7..6599854c 100644 --- a/cps/translations/sv/LC_MESSAGES/messages.po +++ b/cps/translations/sv/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/Calibre-Web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2021-05-13 11:00+0000\n" "Last-Translator: Jonatan Nyberg \n" "Language: sv\n" @@ -16,509 +16,527 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Statistik" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Server startas om, vänligen uppdatera sidan" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Stänger servern, vänligen stäng fönstret" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Okänt kommando" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "Testa e-post i kö för att skicka till %(email)s, vänligen kontrollera Uppgifter för resultat" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Okänd" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Administrationssida" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Grundläggande konfiguration" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Användargränssnitt konfiguration" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "Anpassad kolumn n.%(column)d finns inte i calibre-databasen" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Redigera användare" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Alla" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Användaren hittades inte" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} användare har tagits bort" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Visa alla" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Felaktig begäran" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Gästnamn kan inte ändras" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Gäst kan inte ha den här rollen" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Ingen administratörsanvändare kvar, kan inte ta bort administratörsrollen" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Värdet måste vara sant eller falskt" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Ogiltig roll" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Gästen kan inte ha den här vyn" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "Ogiltig vy" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Gästens språk bestäms automatiskt och kan inte ställas in" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Inget giltigt språk anges" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Inget giltigt bokspråk anges" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Parameter hittades inte" -#: cps/admin.py:572 +#: cps/admin.py:578 #, fuzzy msgid "Invalid Read Column" msgstr "Ogiltig roll" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web konfiguration uppdaterad" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Vill du verkligen ta bort Kobo-token?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Vill du verkligen ta bort den här domänen?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Vill du verkligen ta bort den här användaren?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Är du säker på att du vill ta bort hyllan?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Är du säker på att du vill ändra språk för valda användare?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Är du säker på att du vill ändra synliga bokspråk för valda användare?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Är du säker på att du vill ändra den valda rollen för de valda användarna?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Är du säker på att du vill ändra de valda begränsningarna för de valda användarna?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Är du säker på att du vill ändra de valda synlighetsbegränsningarna för de valda användarna?" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Är du säker på att du vill ändra den valda rollen för de valda användarna?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Är du säker på att du vill stoppa Calibre-Web?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Förneka" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Tillåt" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Taggen hittades inte" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Ogiltig åtgärd" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json är inte konfigurerad för webbapplikation" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Loggfilens plats är inte giltig, vänligen ange rätt sökväg" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "Åtkomstloggplatsens plats är inte giltig, vänligen ange rätt sökväg" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "Vänligen ange en LDAP-leverantör, port, DN och användarobjektidentifierare" -#: cps/admin.py:1211 +#: cps/admin.py:1223 #, fuzzy msgid "Please Enter a LDAP Service Account and Password" msgstr "Ange giltigt användarnamn för att återställa lösenordet" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP-gruppobjektfilter måste ha en \"%s\"-formatidentifierare" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAP-gruppobjektfilter har omatchande parentes" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP-användarobjektfilter måste ha en \"%s\"-formatidentifierare" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAP-användarobjektfilter har omatchad parentes" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "Användarfilter för LDAP-medlemmar måste ha en \"%s\"-formatidentifierare" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "Användarfilter för LDAP-medlemmar har omatchad parentes" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP-certifikat, certifikat eller nyckelplats är inte giltigt, vänligen ange rätt sökväg" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Lägg till ny användare" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Ändra SMTP-inställningar" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Databasfel: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "Testa e-post i kö för att skicka till %(email)s, vänligen kontrollera Uppgifter för resultat" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Det gick inte att skicka Testmeddelandet: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Vänligen konfigurera din e-postadress först..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "E-postserverinställningar uppdaterade" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Ett okänt fel uppstod. Försök igen senare." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Redigera användaren %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "Lösenord för användaren %(user)s återställd" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Konfigurera SMTP-postinställningarna först..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Visaren för loggfil" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Begär uppdateringspaketet" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Hämtar uppdateringspaketet" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Packar upp uppdateringspaketet" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Ersätta filer" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Databasanslutningarna är stängda" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Stoppar server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Uppdatering klar, tryck på okej och uppdatera sidan" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Uppdateringen misslyckades:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP-fel" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Anslutningsfel" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Tiden ute när du etablerade anslutning" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Allmänt fel" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "Uppdateringsfilen kunde inte sparas i Temp Dir" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 #, fuzzy msgid "Failed to extract at least One LDAP User" msgstr "Det gick inte att skapa minst en LDAP-användare" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "Det gick inte att skapa minst en LDAP-användare" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "Fel: %(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "Fel: Ingen användare återges som svar på LDAP-servern" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "Minst en LDAP-användare hittades inte i databasen" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} användare har importerats" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "DB-plats är inte giltig, vänligen ange rätt sökväg" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "DB är inte skrivbar" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "Keyfile-platsen är inte giltig, vänligen ange rätt sökväg" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "Certfile-platsen är inte giltig, vänligen ange rätt sökväg" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "E-postserverinställningar uppdaterade" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Funktion konfiguration" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Fyll i alla fält!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-posten är inte från giltig domän" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Lägg till ny användare" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Användaren '%(user)s' skapad" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Hittade ett befintligt konto för den här e-postadressen eller namnet." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Användaren '%(nick)s' borttagen" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Det går inte att ta bort gästanvändaren" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Ingen adminstratörsanvändare kvar, kan inte ta bort användaren" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Användaren '%(nick)s' uppdaterad" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Sök" + #: cps/converter.py:31 msgid "not installed" msgstr "inte installerad" @@ -527,128 +545,123 @@ msgstr "inte installerad" msgid "Execution permissions missing" msgstr "Körningstillstånd saknas" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "Anpassad kolumn n.%(column)d finns inte i calibre-databasen" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Ingen" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Hoppsan! Vald boktitel är inte tillgänglig. Filen finns inte eller är inte tillgänglig" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "Identifierare är inte skiftlägeskänsliga, skriver över gammal identifierare" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadata uppdaterades" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "Filen %(file)s uppladdad" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Källa eller målformat för konvertering saknas" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "Boken är i kö för konvertering till %(book_format)s" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Det gick inte att konvertera den här boken: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Uppladdad bok finns förmodligen i biblioteket, överväg att ändra innan du laddar upp nya: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Hoppsan! Vald boktitel är inte tillgänglig. Filen finns inte eller är inte tillgänglig" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "Identifierare är inte skiftlägeskänsliga, skriver över gammal identifierare" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s är inte ett giltigt språk" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadata uppdaterades" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Uppladdad bok finns förmodligen i biblioteket, överväg att ändra innan du laddar upp nya: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "Filändelsen '%(ext)s' får inte laddas upp till den här servern" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "Filändelsen '%(ext)s' får inte laddas upp till den här servern" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Filen som ska laddas upp måste ha en ändelse" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "Filen %(filename)s kunde inte sparas i temp dir" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "Det gick inte att flytta omslagsfil %(file)s: %(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "Bokformat har tagits bort" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "Boken har tagits bort" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "redigera metadata" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "Det gick inte att skapa sökväg %(path)s (behörighet nekad)." -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Det gick inte att lagra filen %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "Filformatet %(ext)s lades till %(book)s" @@ -661,487 +674,480 @@ msgstr "Installationen av Google Drive är inte klar, försök att inaktivera oc msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Återuppringningsdomänen är inte verifierad, följ stegen för att verifiera domänen i Google utvecklarkonsol" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(format)s formatet hittades inte för bok-id: %(book)d" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(format)s hittades inte på Google Drive: %(fn)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s hittades inte: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Skicka till Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Detta e-postmeddelande har skickats via Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web test e-post" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Test e-post" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Kom igång med Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Registrera e-post för användare: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Konvertera %(orig)s till %(format)s och skicka till Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Skicka %(format)s till Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Skicka till Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "Den begärda filen kunde inte läsas. Kanske fel behörigheter?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Borttagning av bokmapp för boken %(id)s misslyckades, sökvägen har undermappar: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Borttagning av boken %(id)s misslyckades: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, fuzzy, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "Borttagning av boken %(id)s, boksökväg inte giltig: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Byt namn på titel från: \"%(src)s\" till \"%(dest)s\" misslyckades med fel: %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Filen %(file)s hittades inte på Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Byt namn på titel från: \"%(src)s\" till \"%(dest)s\" misslyckades med fel: %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Boksökvägen %(path)s hittades inte på Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Detta användarnamn är redan taget" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Ogiltigt e-postadressformat" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Fel vid hämtning av omslaget" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Fel på omslagsformat" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Det gick inte att skapa sökväg för omslag" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "Omslagsfilen är inte en giltig bildfil eller kunde inte lagras" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "Endast jpg/jpeg/png/webp/bmp-filer stöds som omslagsfil" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "Endast jpg/jpeg-filer stöds som omslagsfil" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Upptäck" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "UnRar binär fil hittades inte" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "Fel vid körning av UnRar" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "DB är inte skrivbar" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "Körningstillstånd saknas" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "Fel vid körning av UnRar" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "Vänligen få tillgång till calibre-web från icke localhost för att få giltig api_endpoint för Kobo-enhet" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo-installation" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "Registrera dig med %(provider)s" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "du är nu inloggad som: \"%(nickname)s\"" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "Länk till %(oauth)s lyckades" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "Inloggningen misslyckades, ingen användare kopplad till OAuth-konto" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "Sluta länka till %(oauth)s lyckades" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "Sluta länka till %(oauth)s misslyckades" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "Inte länkad till %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Det gick inte att logga in med GitHub." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Det gick inte att hämta användarinformation från GitHub." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Det gick inte att logga in med Google." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Det gick inte att hämta användarinformation från Google." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth-fel, försök igen senare." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "GitHub Oauth-fel: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth-fel, försök igen senare." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google Oauth-fel: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} stjärnor" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Logga in" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token hittades inte" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token har löpt ut" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Lyckades! Vänligen återvänd till din enhet" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Böcker" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Visa senaste böcker" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Heta böcker" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Visa heta böcker" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Hämtade böcker" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Visa hämtade böcker" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Bäst rankade böcker" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Visa böcker med bästa betyg" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Lästa böcker" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Visa lästa och olästa" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Olästa böcker" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Visa olästa" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Upptäck" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Visa slumpmässiga böcker" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategorier" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Visa kategorival" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Serier" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Visa serieval" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Författare" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Visa författarval" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Förlag" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Visa urval av förlag" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Språk" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Visa språkval" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Betyg" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Visa val av betyg" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Filformat" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Visa val av filformat" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Arkiverade böcker" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Visa arkiverade böcker" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Boklista" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Visa boklista" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Sök" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Publicerad efter " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Publicerad före " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Betyg <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Betyg >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Lässtatus = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Avancerad sökning" @@ -1197,7 +1203,7 @@ msgstr "Boken har tagits bort från hyllan: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Skapa en hylla" @@ -1252,45 +1258,45 @@ msgstr "En offentlig hylla med namnet \"%(title)s\" finns redan." msgid "A private shelf with the name '%(title)s' already exists." msgstr "En privat hylla med namnet \"%(title)s\" finns redan." -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Hylla: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Fel vid öppning av hyllan. Hylla finns inte eller är inte tillgänglig" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Uppgifter" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Väntar" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Misslyckades" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Startad" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Klar" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Okänd status" @@ -1323,178 +1329,178 @@ msgstr "En ny uppdatering är tillgänglig. Klicka på knappen nedan för att up msgid "No release information available" msgstr "Ingen versionsinformation tillgänglig" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Upptäck (slumpmässiga böcker)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Heta böcker (mest hämtade)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "Hämtade böcker av %(user)s" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Författare: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Förlag: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Serier: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Betyg: %(rating)s stars" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Filformat: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategori: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Språk: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Hämtningar" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Betygslista" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Lista över filformat" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Konfigurera SMTP-postinställningarna först..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "Boken är i kö för att skicka till %(eReadermail)s" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Det gick inte att skicka den här boken: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "Konfigurera din kindle-e-postadress först..." -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Registrera" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "E-postservern är inte konfigurerad, kontakta din administratör!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "E-postservern är inte konfigurerad, kontakta din administratör!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Din e-post är inte tillåten att registrera" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Bekräftelsemail skickades till ditt e-postkonto." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "Det går inte att aktivera LDAP-autentisering" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "du är nu inloggad som: \"%(nickname)s\"" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Det gick inte att logga in: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Fel användarnamn eller lösenord" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Nytt lösenord skickades till din e-postadress" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Ett okänt fel uppstod. Försök igen senare." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Ange giltigt användarnamn för att återställa lösenordet" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "du är nu inloggad som: \"%(nickname)s\"" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)ss profil" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profilen uppdaterad" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Hittade ett befintligt konto för den här e-postadressen" @@ -1502,54 +1508,58 @@ msgstr "Hittade ett befintligt konto för den här e-postadressen" msgid "Found no valid gmail.json file with OAuth information" msgstr "Hittade ingen giltig gmail.json-fil med OAuth-information" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Skicka till Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "calibre e-bokkonverterings %(tool)s hittades inte" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "%(format)s-format hittades inte på disken" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "E-bokkonverteraren misslyckades med okänt fel" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify-konverteraren misslyckades: %(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "Konverterad fil hittades inte eller mer än en fil i mappen %(folder)s" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "E-bokkonverteraren misslyckades: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "calibre misslyckades med fel: %(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "E-bokkonverteraren misslyckades: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1558,30 +1568,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "redigera metadata" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Ladda upp" @@ -1600,12 +1606,12 @@ msgstr "Smeknamn" msgid "Email" msgstr "E-post" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Administratör" @@ -1615,8 +1621,8 @@ msgstr "Administratör" msgid "Password" msgstr "Lösenord" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Hämta" @@ -1828,13 +1834,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Avbryt" @@ -1884,16 +1890,76 @@ msgstr "Sortera efter publiceringsdatum, nyast först" msgid "Sort according to publishing date, oldest first" msgstr "Sortera efter publiceringsdatum, äldsta först" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "minska" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Mer av" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, fuzzy, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Språk" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Förlag" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Publicerad" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Beskrivning:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Föregående" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Nästa" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Inga resultat hittades" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Hem" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Sök i bibliotek" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Logga ut" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Ta bort boken" @@ -1922,99 +1988,107 @@ msgstr "Konvertera till:" msgid "Convert book" msgstr "Konvertera boken" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Laddar upp..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Stäng" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Fel" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Uppladdning klar, bearbetning, vänligen vänta ..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Ladda upp format" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Boktitel" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Författare" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Taggar" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Serie-ID" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Publiceringsdatum" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Betyg" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Beskrivning" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Identifierare" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Identifierartyp" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Identifierarvärde" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Ta bort" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Lägg till identifierare" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Taggar" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Serie-ID" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Betyg" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Omslagswebbadress (jpg, omslag hämtas och lagras i databasen, fältet är efteråt tomt igen)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Ladda upp omslag från lokal enhet" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Publiceringsdatum" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Förlag" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Språk" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Ja" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Nej" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Ladda upp format" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Visa bok vid Spara" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Hämta metadata" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2022,38 +2096,32 @@ msgstr "Hämta metadata" msgid "Save" msgstr "Spara" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Sökord" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Sök sökord " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Klicka på omslaget för att läsa in metadata till formuläret" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Läser in..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Stäng" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Källa" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Sökningsfel!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Inga resultat hittades! Försök med ett annat sökord." @@ -2162,7 +2230,7 @@ msgid "Enter " msgstr "Identifierare" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Är du verkligen säker?" @@ -2641,74 +2709,61 @@ msgstr "Lägg till tillåtna/avvisade taggar" msgid "Add Allowed/Denied custom column values" msgstr "Lägg till tillåtna/avvisade anpassade kolumnvärden" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Läs i webbläsaren" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Lyssna i webbläsaren" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, fuzzy, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Publicerad" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Markera som oläst" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Markera som läst" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Markera som oläst" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Läst" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Återställ från arkivet" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Lägg till i arkivet" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Arkiverad" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Beskrivning:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Lägg till hyllan" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Publik)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Redigera metadata" @@ -2781,10 +2836,6 @@ msgstr "Ange domännamn" msgid "Denied Domains (Blacklist)" msgstr "Avvisade domäner för registrering" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Nästa" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "Öppna filen .kobo/Kobo/Kobo eReader.conf i en textredigerare och lägg till (eller redigera):" @@ -2807,11 +2858,16 @@ msgstr "E-postservern är inte konfigurerad, kontakta din administratör!" msgid "Create Issue" msgstr "Skapa ärende" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Funktion konfiguration" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Tillbaka till hemmet" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2901,7 +2957,7 @@ msgstr "Böcker sorterade efter Betyg" msgid "Books ordered by file formats" msgstr "Böcker ordnade av filformat" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Hyllor" @@ -2910,60 +2966,37 @@ msgstr "Hyllor" msgid "Books organized in shelves" msgstr "Böcker organiserade i hyllor" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Hem" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Växla navigering" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Sök i bibliotek" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "Enkel" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Konto" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Logga ut" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Laddar upp..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Fel" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Uppladdning klar, bearbetning, vänligen vänta ..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Inställningar" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Vänligen uppdatera inte sidan" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Bläddra" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Om" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Föregående" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Bokdetaljer" @@ -3079,7 +3112,7 @@ msgstr "Föräldramapp" msgid "Select" msgstr "Välj" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3087,36 +3120,82 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "Calibre-Web e-bokkatalog" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "PDF-läsare" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Välj ett användarnamn" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Ljust" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Mörkt" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Tillbaka" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Fyll i texten igen när sidofält är öppna." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Ta bort" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Väntar" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Vertikal" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Läst" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Ogiltig roll" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3298,10 +3377,6 @@ msgstr "Länken går ut efter 10 minuter." msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Inga resultat hittades" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Sökterm:" @@ -3318,11 +3393,11 @@ msgstr "Publiceringsdatum från" msgid "Published Date To" msgstr "Publiceringsdatum till" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3358,11 +3433,13 @@ msgstr "Betyg större än" msgid "Rating Below" msgstr "Betyg mindre än" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3386,6 +3463,16 @@ msgstr "Inaktivera ändring av ordning" msgid "Enable Change order" msgstr "Aktivera ändring av ordning" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "Sortera efter bokdatum, nyast först" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "Sortera efter bokdatum, äldsta först" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Dela med alla" @@ -3454,15 +3541,20 @@ msgstr "Förlopp" msgid "Run Time" msgstr "Drifttid" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Slå samman" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3470,6 +3562,10 @@ msgstr "" msgid "Reset user Password" msgstr "Återställ användarlösenordet" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Visa böcker med språk" diff --git a/cps/translations/tr/LC_MESSAGES/messages.mo b/cps/translations/tr/LC_MESSAGES/messages.mo index 97cc6264..c0cec28d 100644 Binary files a/cps/translations/tr/LC_MESSAGES/messages.mo and b/cps/translations/tr/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/tr/LC_MESSAGES/messages.po b/cps/translations/tr/LC_MESSAGES/messages.po index f93f6d18..e32f8d11 100644 --- a/cps/translations/tr/LC_MESSAGES/messages.po +++ b/cps/translations/tr/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2020-04-23 22:47+0300\n" "Last-Translator: iz \n" "Language: tr\n" @@ -16,506 +16,524 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "İstatistikler" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Sunucu yeniden başlatıldı, lütfen sayfayı yeniden yükleyin" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Sunucu kapatıyor, lütfen pencereyi kapatın" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "%(eReadermail)s'a gönderilmek üzere başarıyla sıraya alındı" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Bilinmeyen" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Yönetim sayfası" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Temel Ayarlar" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Arayüz Ayarları" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Tümü" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web yapılandırması güncellendi" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 #, fuzzy msgid "Please Enter a LDAP Service Account and Password" msgstr "Şifrenizi sıfırlayabilmek için lütfen geçerli bir kullanıcı adı giriniz" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "Deneme e-postası gönderilirken bir hata oluştu: %(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "Lütfen önce e-posta adresinizi ayarlayın..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "E-posta sunucusu ayarları güncellendi" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Bilinmeyen bir hata oluştu. Lütfen daha sonra tekrar deneyiniz." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "%(nick)s kullanıcısını düzenle" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "%(user)s kullanıcısının şifresi sıfırlandı" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Lütfen önce SMTP e-posta ayarlarını ayarlayın..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "Log dosyası görüntüleyici" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Güncelleme paketi isteniyor" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Güncelleme paketi indiriliyor" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Güncelleme paketi ayıklanıyor" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Dosyalar değiştiriliyor" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Veritabanı bağlantıları kapalı" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Sunucu durduruyor" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Güncelleme tamamlandı, sayfayı yenilemek için lütfen Tamam'a tıklayınız" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Güncelleme başarısız:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP Hatası" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Bağlantı hatası" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "Bağlantı kurulmaya çalışırken zaman aşımına uğradı" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Genel hata" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "%(filename)s dosyası geçici dizine kaydedilemedi" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "E-posta sunucusu ayarları güncellendi" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Özellik Yapılandırması" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Lütfen tüm alanları doldurun!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "E-posta izin verilen bir servisten değil" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Yeni kullanıcı ekle" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "'%(user)s' kullanıcısı oluşturuldu" -#: cps/admin.py:1949 +#: cps/admin.py:1972 #, fuzzy msgid "Oops! An account already exists for this Email. or name." msgstr "Bu e-posta adresi veya kullanıcı adı için zaten bir hesap var." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Kullanıcı '%(nick)s' silindi" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "Başka yönetici kullanıcı olmadığından silinemedi" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "'%(nick)s' kullanıcısı güncellendi" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Ara" + #: cps/converter.py:31 msgid "not installed" msgstr "yüklü değil" @@ -524,128 +542,123 @@ msgstr "yüklü değil" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Hiçbiri" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metaveri başarıyla güncellendi" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "%(file)s dosyası yüklendi" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Dönüştürme için kaynak ya da hedef biçimi eksik" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "eKitap %(book_format)s formatlarına dönüştürülmek üzere başarıyla sıraya alındı" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Bu eKitabı dönüştürürken bir hata oluştu: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "Yüklenen eKitap muhtemelen kitaplıkta zaten var. Yenisini yüklemeden değiştirmeyi düşünün: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s geçerli bir dil değil" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metaveri başarıyla güncellendi" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "Yüklenen eKitap muhtemelen kitaplıkta zaten var. Yenisini yüklemeden değiştirmeyi düşünün: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "'%(ext)s' uzantılı dosyaların bu sunucuya yüklenmesine izin verilmiyor" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "'%(ext)s' uzantılı dosyaların bu sunucuya yüklenmesine izin verilmiyor" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Yüklenecek dosyanın mutlaka bir uzantısı olması gerekli" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "%(filename)s dosyası geçici dizine kaydedilemedi" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "metaveri düzenle" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "%(path)s dizini oluşturulamadı. (İzin reddedildi)" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "%(file)s dosyası kaydedilemedi." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "%(book)s kitabına %(ext)s dosya biçimi eklendi" @@ -658,482 +671,475 @@ msgstr "Google Drive kurulumu tamamlanmadı, Google Drive'ı devre dışı bıra msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Geri yönlendirme alanı (callback domain) doğrulanamadı, lütfen Google geliştirici konsolunda alan adını doğrulamak için gerekli adımları izleyin." -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "%(book)d nolu kitap için %(format)s biçimi bulunamadı" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "%(fn)s eKitabı için %(format)s biçimi Google Drive'da bulunamadı" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s bulunamadı: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Kindle'a gönder" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "Bu e-Posta Calibre-Web ile gönderilmiştir." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web deneme e-Postası" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Deneme e-Postası" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Calibre-Web'i Kullanmaya Başlayın" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Kullanıcı Kayıt e-Postası: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "%(orig)s'dan %(format)s biçimine çevir ve Kindle'a gönder" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "%(format)s biçimlerini Kindle'a gönder" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Kindle'a gönder" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "İstenilen dosya okunamadı. Yanlış izinlerden kaynaklanabilir?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Kitap adını değiştirme sırasında hata oluştu ('%(src)s' → '%(dest)s'): %(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "%(file)s dosyası Google Drive'da bulunamadı" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "Kitap adını değiştirme sırasında hata oluştu ('%(src)s' → '%(dest)s'): %(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "eKitap yolu %(path)s Google Drive'da bulunamadı" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Bu kullanıcı adı zaten alındı" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Keşfet" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "%(provider)s ile Kaydol" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "giriş yaptınız: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "GitHub ile giriş yapılamadı." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Github'dan kullanıcı bilgileri alınamadı." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Google ile giriş yapılamadı." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Google'dan kullanıcı bilgileri alınamadı." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth hatası, lütfen tekrar deneyin." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth hatası, lütfen tekrar deneyin." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Giriş" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Token bulunamadı" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token süresi doldu" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Başarılı! Lütfen cihazınıza dönün" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "eKitaplar" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Son eKitapları göster" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Popüler" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Okunanlar" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Okunan ve okunmayanları göster" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Okunmamışlar" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Okunmamışları göster" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Keşfet" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Rastgele Kitap Göster" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Kategoriler" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Kategori seçimini göster" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Seriler" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Seri seçimini göster" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Yazarlar" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Yazar seçimini göster" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Yayıncılar" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Yayıncı seçimini göster" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Diller" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Dil seçimini göster" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Değerlendirmeler" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Değerlendirme seçimini göster" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Biçimler" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Dosya biçimi seçimini göster" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Son eKitapları göster" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Ara" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Yayınlanma (sonra)" -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Yayınlanma (önce)" -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Değerlendirme <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Değerlendirme >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Gelişmiş Arama" @@ -1189,7 +1195,7 @@ msgstr "eKitap kitaplıktan silindi: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Kitaplık oluştur" @@ -1244,45 +1250,45 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Kitaplık: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Kitaplık açılırken hata oluştu. Kitaplık mevcut değil ya da erişilebilir değil" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Görevler" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Bekleniyor" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Başarısız" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Başladı" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Bitti" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Bilinmeyen Durum" @@ -1315,177 +1321,177 @@ msgstr "Yeni bir güncelleme mevcut. Son sürüme güncellemek için aşağıdak msgid "No release information available" msgstr "Sürüm bilgisi mevcut değil" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Keşfet (Rastgele)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "Yazar: %(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Yayınevi: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Seri: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Değerlendirme: %(rating)s yıldız" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Biçim: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Kategori: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Dil: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Değerlendirme listesi" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Biçim listesi" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Lütfen önce SMTP e-posta ayarlarını ayarlayın..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "%(eReadermail)s'a gönderilmek üzere başarıyla sıraya alındı" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Kayıt ol" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "E-Posta sunucusu ayarlanmadı, lütfen yöneticinizle iletişime geçin!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "E-Posta sunucusu ayarlanmadı, lütfen yöneticinizle iletişime geçin!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "E-posta adresinizle kaydolunmasına izin verilmiyor" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "Onay e-Postası hesabınıza gönderildi." -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "LDAP Kimlik Doğrulaması etkinleştirilemiyor" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "giriş yaptınız: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Yanlış Kullanıcı adı ya da Şifre" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Yeni şifre e-Posta adresinize gönderildi" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Bilinmeyen bir hata oluştu. Lütfen daha sonra tekrar deneyiniz." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Şifrenizi sıfırlayabilmek için lütfen geçerli bir kullanıcı adı giriniz" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "giriş yaptınız: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s Profili" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Profil güncellendi" -#: cps/web.py:1515 +#: cps/web.py:1530 #, fuzzy msgid "Oops! An account already exists for this Email." msgstr "Bu e-posta adresi için bir hesap mevcut." @@ -1494,54 +1500,58 @@ msgstr "Bu e-posta adresi için bir hesap mevcut." msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Kindle'a gönder" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "eKitap-Dönüştürücü hatası: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "eKitap-Dönüştürücü hatası: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1550,30 +1560,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "metaveri düzenle" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Yükleme" @@ -1592,12 +1598,12 @@ msgstr "Kullanıcı adı" msgid "Email" msgstr "" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "E-Posta adresiniz" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Yönetim" @@ -1607,8 +1613,8 @@ msgstr "Yönetim" msgid "Password" msgstr "Şifre" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "İndirme" @@ -1819,13 +1825,13 @@ msgid "OK" msgstr "" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "" @@ -1875,16 +1881,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "azalt" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "daha fazla" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Dil" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Yayınevi" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Açıklama:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Önceki" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Sonraki" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Anasayfa" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Çıkış" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "eKitabı Sil" @@ -1913,99 +1979,107 @@ msgstr "Dönüştür:" msgid "Convert book" msgstr "eKitabı dönüştür" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Yükleniyor..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Kapak" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Hata" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Yükleme tamamlandı, işleniyor, lütfen bekleyin..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Kitap Adı" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Yazar" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Etiketler" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Değerlendirme" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Açıklama" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Etiketler" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Değerlendirme" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Yayınevi" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Dil" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Evet" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Hayır" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2013,38 +2087,32 @@ msgstr "" msgid "Save" msgstr "" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Anahtar Kelime" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr "Anahtar kelime ara" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Forma metaveri yüklemek için kapağa tıklayın" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Yükleniyor..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Kapak" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Kaynak" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Arama hatası!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "" @@ -2153,7 +2221,7 @@ msgid "Enter " msgstr "Kayıt ol" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Emin misiniz?" @@ -2629,74 +2697,61 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Okunmadı olarak işaretle" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Okundu olarak işaretle" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Okunmadı olarak işaretle" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Okudum" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Ara" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Açıklama:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Kitaplığa ekle" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "" @@ -2767,10 +2822,6 @@ msgstr "Servis adı girin" msgid "Denied Domains (Blacklist)" msgstr "" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Sonraki" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2792,11 +2843,16 @@ msgstr "E-Posta sunucusu ayarlanmadı, lütfen yöneticinizle iletişime geçin! msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Özellik Yapılandırması" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2886,7 +2942,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "Biçime göre sıralanmış eKitaplar" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2895,60 +2951,36 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Anasayfa" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "" -#: cps/templates/layout.html:47 -msgid "Search Library" +#: cps/templates/layout.html:59 +msgid "Simple Theme" msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Hesap" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Çıkış" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Yükleniyor..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Hata" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Yükleme tamamlandı, işleniyor, lütfen bekleyin..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Ayarlar" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Gözat" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Hakkında" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Önceki" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "eKitap Detayları" @@ -3064,7 +3096,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Kitap" @@ -3073,36 +3105,81 @@ msgstr "Kitap" msgid "Calibre-Web eBook Catalog" msgstr "" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 #, fuzzy msgid "epub Reader" msgstr "PDF Okuyucu" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Kullanıcı adı seç" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Açık" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Koyu" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Geri" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Kenar çubukları açıkken metni kaydır" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Sil" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Bekleniyor" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Dikey" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Okudum" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 #, fuzzy msgid "Comic Reader" @@ -3284,10 +3361,6 @@ msgstr "" msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "" @@ -3304,11 +3377,11 @@ msgstr "" msgid "Published Date To" msgstr "" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3345,11 +3418,13 @@ msgstr "" msgid "Rating Below" msgstr "" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3373,6 +3448,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "" @@ -3441,15 +3524,19 @@ msgstr "İlerleme" msgid "Run Time" msgstr "" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3457,6 +3544,10 @@ msgstr "" msgid "Reset user Password" msgstr "Kullanıcı şifresini sıfırla" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "" diff --git a/cps/translations/uk/LC_MESSAGES/messages.mo b/cps/translations/uk/LC_MESSAGES/messages.mo index 4b76e65c..34f2d302 100644 Binary files a/cps/translations/uk/LC_MESSAGES/messages.mo and b/cps/translations/uk/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/uk/LC_MESSAGES/messages.po b/cps/translations/uk/LC_MESSAGES/messages.po index bbce3961..7de10c3d 100644 --- a/cps/translations/uk/LC_MESSAGES/messages.po +++ b/cps/translations/uk/LC_MESSAGES/messages.po @@ -6,7 +6,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/calibre-web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2017-04-30 00:47+0300\n" "Last-Translator: ABIS Team \n" "Language: uk\n" @@ -15,507 +15,525 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Статистика" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Сервер перезавантажено, будь-ласка, перезавантажте сторінку" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Виконується зупинка серверу, будь-ласка, закрийте вікно" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Невідома команда" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Невідомий" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Сторінка адміністратора" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Настройки сервера" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Конфігурація інтерфейсу" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 #, fuzzy msgid "Edit Users" msgstr "Керування сервером" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Всі" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Користувача не знайдено" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} користувачі видалені успішно" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Показати всі" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Не правильна роль" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Ви справді хочете видалити книжкову полицю?" -#: cps/admin.py:618 +#: cps/admin.py:624 #, fuzzy msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Ви справді хочете видалити книжкову полицю?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 #, fuzzy msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Ви справді хочете видалити книжкову полицю?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 #, fuzzy msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Ви справді хочете видалити книжкову полицю?" -#: cps/admin.py:631 +#: cps/admin.py:637 #, fuzzy msgid "Are you sure you want to change Calibre library location?" msgstr "Ви справді хочете видалити книжкову полицю?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Заборонити" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Дозволити" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Тег не знайдено" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "Змінити налаштування SMTP" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Змінити користувача %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "Будь-ласка, спочатку сконфігуруйте параметри SMTP" -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "Перевірка оновлень" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "Завантаження оновлень" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Розпакування оновлення" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "Заміна файлів" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "З'єднання з базою даних закрите" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Сервер зупиняється" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Оновлення встановлені, натисніть ok і перезавантажте сторінку" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Оновлення неуспішне:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP помилка" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Помилка зʼєднання" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Помилка" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "З'єднання з базою даних закрите" -#: cps/admin.py:1902 +#: cps/admin.py:1925 #, fuzzy msgid "Database Configuration" msgstr "Особливі налаштування" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Будь-ласка, заповніть всі поля!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Додати користувача" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Користувач '%(user)s' додан" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "" -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Користувача '%(nick)s' видалено" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Користувача '%(nick)s' оновлено" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Пошук" + #: cps/converter.py:31 msgid "not installed" msgstr "не встановлено" @@ -524,127 +542,122 @@ msgstr "не встановлено" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "Ні" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "Неможливо відкрити книгу. Файл не існує або немає доступу." - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "Неможливо відкрити книгу. Файл не існує або немає доступу." + +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" msgstr "" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "Завантажувальний файл повинен мати розширення" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "змінити метадані" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "" -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "" @@ -657,479 +670,472 @@ msgstr "" msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "Домен зворотнього зв'язку не підтверджено. Виконайте дії для підтвердження домену, будь-ласка" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Відправити на Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "" -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Надіслати %(format)s до E-Reader" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "Відправити на Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Огляд" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "Ви увійшли як користувач: '%(nickname)s'" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} зірок" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Ім'я користувача" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Токен не знайдено" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Час дії токено вичерпано" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Вдалося! Будь-ласка, поверніться до вашого пристрою" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Книжки" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Показувати останні книги" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Популярні книги" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Показувати популярні книги" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Книги з найкращим рейтингом" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Показувати книги з найвищим рейтингом" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Прочитані книги" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Показувати прочитані та непрочитані книги" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Непрочитані книги" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Показати не прочитані" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Огляд" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Показувати випадкові книги" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Категорії" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Показувати вибір категорії" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Серії" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Показувати вибір серії" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Автори" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Показувати вибір автора" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Видавництва" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Показувати вибір серії" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Мови" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Показувати вибір мови" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Рейтинги" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Показувати вибір серії" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Формати файлів" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Показувати вибір серії" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Архівні книжки" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Архівні книжки" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Список книжок" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Пошук" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Опубліковані після " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Опубліковано до " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Рейтинг <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Рейтинг >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Розширений пошук" @@ -1185,7 +1191,7 @@ msgstr "Книга видалена з книжкової полиці: %(sname) msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "створити книжкову полицю" @@ -1239,45 +1245,45 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "Книжкова полиця: '%(name)s'" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "Помилка при відкриванні полиці. Полиця не існує або до неї відсутній доступ" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Завдання" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Очікує" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Розпочато" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Завершено" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "Закінчено" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "Відмінено" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Невідомий статус" @@ -1310,173 +1316,173 @@ msgstr "" msgid "No release information available" msgstr "" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "Огляд (випадкові книги)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "Популярні книги (найбільш завантажувані)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "Видавництво: %(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "Серії: %(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "Рейтинг: Відсутній" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "Рейтинг: %(rating)s зірок" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "Формат файлу: %(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "Категорія: %(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "Мова: %(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Завантаження" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Список рейтингів" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Список форматів файлу" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "Будь-ласка, спочатку сконфігуруйте параметри SMTP" -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "Помилка при відправці книги: %(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Зареєструватись" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "Ви увійшли як користувач: '%(nickname)s'" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Помилка в імені користувача або паролі" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "" -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Помилка в імені користувача або паролі" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "Ви увійшли як користувач: '%(nickname)s'" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "Профіль %(name)s" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Профіль оновлено" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "" @@ -1484,54 +1490,58 @@ msgstr "" msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "Відправити на Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1540,30 +1550,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "змінити метадані" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Додати нову книгу" @@ -1582,12 +1588,12 @@ msgstr "Ім'я користувача" msgid "Email" msgstr "" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Адмін" @@ -1597,8 +1603,8 @@ msgstr "Адмін" msgid "Password" msgstr "Пароль" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Завантажити" @@ -1809,13 +1815,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Скасувати" @@ -1865,16 +1871,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Більше за" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Мова" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Видавець" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Опубліковано" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Опис:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Попередній перегляд" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Далі" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Вийти" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Видалити книгу" @@ -1903,99 +1969,107 @@ msgstr "Конвертувати в:" msgid "Convert book" msgstr "Конвертувати книгу" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Завантаження..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Закрити" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Помилка" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "" + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Формат завантаження" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Назва книги" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Автор" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Теги" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Опубліковано" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Рейтинг" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Опис" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Ідентифікатори" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Видалити" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Додати ідентифікатор" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Теги" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Рейтинг" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Опубліковано" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Видавець" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Мова" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Так" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Ні" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Формат завантаження" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "переглянути книгу після редагування" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Отримати метадані" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2003,38 +2077,32 @@ msgstr "Отримати метадані" msgid "Save" msgstr "Зберегти" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Ключове слово" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " Пошук по ключовому слову" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Натисніть на обкладинку, щоб отримати метадані" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Завантаження..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Закрити" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Джерело" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Помилка пошуку!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "" @@ -2142,7 +2210,7 @@ msgid "Enter " msgstr "Зареєструватись" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "" @@ -2618,74 +2686,61 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Відкрити в браузері" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Прослухати в браузері" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Опубліковано" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Відмітити як не прочитане" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Відмітити як прочитане" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Відмітити як не прочитане" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Прочитано" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Відновити з архіву" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Додати в архів" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Заархівовано" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Опис:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Додати на книжкову полицю" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Публічно)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Редагувати метадані" @@ -2756,10 +2811,6 @@ msgstr "Введіть домен" msgid "Denied Domains (Blacklist)" msgstr "Заборонені домени (Чорний список)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Далі" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2780,11 +2831,16 @@ msgstr "" msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Особливі налаштування" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2874,7 +2930,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2883,60 +2939,36 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "Включити навігацію" -#: cps/templates/layout.html:47 -msgid "Search Library" +#: cps/templates/layout.html:59 +msgid "Simple Theme" msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Вийти" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Завантаження..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Помилка" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "" - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Налаштування" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Перегляд" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "Про програму" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Попередній перегляд" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Деталі" @@ -3052,7 +3084,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 #, fuzzy msgid "Ok" msgstr "Книга" @@ -3061,35 +3093,80 @@ msgstr "Книга" msgid "Calibre-Web eBook Catalog" msgstr "" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Виберіть ім'я користувача" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Назад" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "Переформатувати текст, коли відкриті бічні панелі." -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Видалити" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Очікує" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Вертикально" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Прочитано" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "" @@ -3268,10 +3345,6 @@ msgstr "" msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "" @@ -3288,11 +3361,11 @@ msgstr "Дата публікації з" msgid "Published Date To" msgstr "Дата публікації до" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3329,11 +3402,13 @@ msgstr "" msgid "Rating Below" msgstr "" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Від:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "До:" @@ -3357,6 +3432,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "зробити книжкову полицю доступною для всіх?" @@ -3425,15 +3508,20 @@ msgstr "Прогрес" msgid "Run Time" msgstr "Тривалість" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Обʼєднати" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "Дії" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3441,6 +3529,10 @@ msgstr "" msgid "Reset user Password" msgstr "Скинути пароль користувача" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Показувати книги на мовах" diff --git a/cps/translations/vi/LC_MESSAGES/messages.mo b/cps/translations/vi/LC_MESSAGES/messages.mo index ca2423a4..bd2277ef 100644 Binary files a/cps/translations/vi/LC_MESSAGES/messages.mo and b/cps/translations/vi/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/vi/LC_MESSAGES/messages.po b/cps/translations/vi/LC_MESSAGES/messages.po index 3fba0170..b12d1981 100644 --- a/cps/translations/vi/LC_MESSAGES/messages.po +++ b/cps/translations/vi/LC_MESSAGES/messages.po @@ -4,7 +4,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-web\n" "Report-Msgid-Bugs-To: https://github.com/janeczku/calibre-web\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2022-09-20 21:36+0700\n" "Last-Translator: Ha Link \n" "Language: vi\n" @@ -13,500 +13,518 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "Thống kê" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "Máy chủ đã khởi động lại,hãy tải lại trang" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "Máy chủ đang được tắt, hãy đóng cửa sổ này" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "Lệnh không tồn tại" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "Không rõ" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "Trang admin" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "Thiết lập cơ bản" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "Thiết lập UI" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "Chỉnh sửa người dùng" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "Tất cả" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "Không tìm thấy user" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "{} người dung đã đươc xoá thành công" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "Hiển thị tất cả" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "Yêu cầu không đúng định dạng" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "Tên người dung khách không thể thay đổi" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "Khách không thể mang vai trò này" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "Không còn người dùng quản trị, không thể xoá vai trò admin" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "Giá trị phải là true hoặc false" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "Vai trò không hợp lệ" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "Tài khoản khách không có màn hình này" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "View không hợp lệ" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "Ngôn ngữ của khách được xác định tự động và không thể đặt được" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "Địa chỉ cung cấp không hợp lệ" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "Ngôn ngữ sách không hợp lệ" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "Tham số không tồn tại" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "Cột đọc không hợp lệ" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "Cột bị hạn chế không hợp lệ" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Thiết lập Calibre-Web đã cập nhật" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "Bạn có thực sự muốn xóa Kobo Token không?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "Bạn có thực sự muốn xóa miền này không?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "Bạn có thực sự muốn xóa người dùng này không?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "Bạn có chắc chắn muốn xóa giá này không?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "Bạn có chắc chắn muốn thay đổi ngôn ngữ của những người dùng đã chọn?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "Bạn có chắc chắn muốn thay đổi ngôn ngữ sách hiển thị cho (những) người dùng đã chọn không?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "Bạn có chắc chắn muốn thay đổi vai trò đã chọn cho (những) người dùng đã chọn không?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "Bạn có chắc chắn muốn thay đổi những giới hạn đã chọn cho người dung?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "Bạn có chắc chắn muốn thay đổi các giới hạn hiển thị đã chọn cho (những) người dùng đã chọn không?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "Bạn có chắc chắn muốn thay đổi hành vi đồng bộ hóa giá cho (những) người dùng đã chọn không?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "Bạn có chắc chắn muốn thay đổi vị trí thư viện Calibre không?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "Bạn có chắc chắn muốn xóa cơ sở dữ liệu đồng bộ của Calibre-Web để bắt buộc đồng bộ hóa hoàn toàn với Kobo Reader của mình không?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "Từ chối" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "Cho phép" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} mục nhập đồng bộ hóa đã bị xóa" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "Tag không tồn tại" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "Hành động không hợp lệ" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "Vị trí tệp nhật ký không hợp lệ, vui lòng nhập đường dẫn chính xác" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "Thêm người dùng mới" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "Lỗi cơ sở dữ liệu: %(error)s." -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 #, fuzzy msgid "Scheduled tasks settings updated" msgstr "Thiết lập cơ sở dữ lieu đã được cập nhật" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "Lỗi không xác định xảy ra. Xin hãy thử lại sau." -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "Chỉnh sửa người dùng %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "" -#: cps/admin.py:1451 +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." msgstr "" -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "Đang giải nén tệp cập nhật" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "They thế tập tin" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "Liên kết cơ sở dữ liệu đã được đóng" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "Đang dừng server" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "Cập nhật hoàn tất, ấn okay và tải lại trang" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "Cập nhật thất bại:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "Lỗi HTTP" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "Lỗi kết nối" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "Lỗi chung" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "File cập nhật không thể được lưu trong thư mục tạm thời" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "Thiết lập cơ sở dữ lieu đã được cập nhật" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "Thiết lập cơ sở dữ lieu :)))" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "Hãy điền hết các trường!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "Địa chỉ email không hợp lệ" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "Thêm người dùng mới" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "Người dùng '%(user)s' đã được tạo" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "Người dùng với địa chỉ email hoặc tên đã tồn tại." -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "Người dùng '%(nick)s' đã bị xoá" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "Không thể xoá người dùng khách" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "Người dùng '%(nick)s' đã được cập nhật" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "Tìm kiếm" + #: cps/converter.py:31 msgid "not installed" msgstr "chưa cài đặt" @@ -515,127 +533,122 @@ msgstr "chưa cài đặt" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "None" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "Metadata đã được cập nhật thành công" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "File %(file)s đã được tải lên" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "Thiếu định dạng nguồn hoặc đích để convert" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "Có lỗi xảy ra khi chuyển đổi định dạng cho sach: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s không phải là ngôn ngữ hợp lệ" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "Metadata đã được cập nhật thành công" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" msgstr "" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "Lưu file thất bại %(file)s." -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "" @@ -648,483 +661,476 @@ msgstr "" msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "%(format)s không tìm thấy: %(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "Gửi tới Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "E-mail này được gửi từ Calibre-Web." -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web test e-mail" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "Test e-mail" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "Bắt đầu với Calibre-Web" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "Email đăng ký cho người dùng: %(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "Chuyển đổi %(orig)s thành %(format)s và chuyển tới Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "Gửi %(format)s tới Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s gửi tới Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "File được yêu cầu không thể đọc. Có thể do phân quyền bị sai?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "Xoá thư mục cho sách %(id)s thất bại, đường dẫn có subfolders: %(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "Xoá sách %(id)s thất bại: %(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "File %(file)s không tìm thấy trẻn Google Drive" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Không tìm thấy được dẫn sách %(path)s trên Google Drive" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "Username này đã bị sử dụng" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "Định dạng email address không hợp lệ" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "Lỗi tải xuống ảnh bìa" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "Định dạng ảnh bìa lỗi" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "Tạo đường dẫn cho ảnh bìa thất bại" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "Khám phá" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Thiết lập Kobo" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "Đăng nhập với Github thất bại." -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "Lấy thông tin user từ Github thất bại." -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "Đăng nhập với Google thất bại." -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "Lấy thông tin người dùng từ Google thất bại." -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "Oauth Github lỗi, xin thử lại sau." -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "Github Oauth lỗi: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth lỗi, làm ơn thử lại sau." -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google Oauth lỗi: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} sao" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "Đăng nhập" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "Không tìm thấy token" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token đã hết hạn" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "Thành công! Hãy quay lại thiết bị của bạn" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "Sách" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "Hiển thị sách gần đây" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "Sách hot" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "Hiển thị sách hot" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "Sách đã tải" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "Hiển thị sách đã tải" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "Top sách được đánh giá cao" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "Hiện thị top những sách được đánh giá cao" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "Sách đã đọc" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "Hiển thị đã đọc và chưa đọc" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "Sách chưa đọc" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "Hiện thị sách chưa đọc" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "Khám phá" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "Hiển thị sách ngẫu nhiên" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "Chủ đề" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "Hiển thị chọn chủ đề" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "Series" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "Hiển thị chọn series" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "Tác giả" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "Hiển thị chọn tác giả" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "Nhà phát hành" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "Hiển thị chọn nhà phát hành" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "Ngôn ngữ" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "Hiển thị chọn ngôn ngữ" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "Đánh giá" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "Hiển thị chọn đánh giá" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "Định dạng file" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "Hiển thị lựa chọn định dạng file" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "Sách đã lưu trữ" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "Hiện thị sách trong kho" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "Danh sách sách" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "Hiển thị danh sách sách" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "Tìm kiếm" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "Phát hành sau " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "Phát hành trước " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "Đánh giá <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "Đánh giá >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "Trạng thái đọc = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "Tìm kiếm nâng cao" @@ -1179,7 +1185,7 @@ msgstr "Sách đã được xoá khỏi giá: %(sname)s" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "Xin lỗi bạn không có quyền xoá sách khỏi giá này" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "Tạo một giá sách" @@ -1232,46 +1238,46 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "Nhiệm vụ" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "Đang chờ" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "Thất bại" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "Đã bắt đầu" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "Đã hoàn thành" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 #, fuzzy msgid "Cancelled" msgstr "Huỷ bỏ" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "Trạng thái không xác định" @@ -1304,175 +1310,175 @@ msgstr "" msgid "No release information available" msgstr "" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "" -#: cps/web.py:623 +#: cps/web.py:629 #, fuzzy msgid "Rating: None" msgstr "Đánh giá cao hơn" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "Tải xuống" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "Danh sách đánh giá" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "Danh sách định dạng file" -#: cps/web.py:1249 +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." msgstr "" -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "Đăng ký" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "Email của bạn không được cho phép để đăng ký" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "Không thể đăng nhập: %(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "Tên đăng nhập hoặc mật khẩu không đúng" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "Mật khẩu mới đã được gửi đến email của bạn" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "Lỗi không xác định xảy ra. Xin hãy thử lại sau." -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "Tên đăng nhập hoặc mật khẩu không đúng" -#: cps/web.py:1437 +#: cps/web.py:1445 #, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "Metadata đã được cập nhật thành công" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "Tìm thấy một tài khoản đã toàn tại cho địa chỉ email này" @@ -1480,56 +1486,60 @@ msgstr "Tìm thấy một tài khoản đã toàn tại cho địa chỉ email n msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s gửi tới Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "" + +#: cps/tasks/convert.py:345 #, fuzzy msgid "Convert" msgstr "Khám phá" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 #, fuzzy msgid "Reconnecting Calibre database" msgstr "Kết nối lại với cơ sở dữ liệu Calibre" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 #, fuzzy msgid "E-mail" msgstr "Test e-mail" @@ -1538,30 +1548,26 @@ msgstr "Test e-mail" msgid "Backing up Metadata" msgstr "" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "Tải lên" @@ -1580,12 +1586,12 @@ msgstr "Tên người dùng" msgid "Email" msgstr "Địa chỉ email" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "Kindle" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "Admin" @@ -1595,8 +1601,8 @@ msgstr "Admin" msgid "Password" msgstr "Mật khẩu" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "Tải xuống" @@ -1809,13 +1815,13 @@ msgid "OK" msgstr "Ok" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "Huỷ bỏ" @@ -1865,16 +1871,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "Cùng tác giả" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "Sách %(index)s của %(range)s" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "Ngôn ngữ" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "Nhà xuất bản" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "Đã phát hành" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "Mô tả:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "Trước" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "Tiếp tục" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "Không tìm thấy kết quả" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "Trang chủ" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "Tìm kiếm thư viện" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "Đăng xuất" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "Xoá sách" @@ -1903,99 +1969,107 @@ msgstr "Chuyển đổi tới:" msgid "Convert book" msgstr "Chuyển đổi sách" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "Đang tải lên..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "Đóng" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "Lỗi" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "Tải lên xong, đang xử lý, đợi chút ..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "Định dạng tải lên" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "Tên sách" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "Tác giả" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "Đánh dấu" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "Series ID" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "Ngày xuất bản" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "Đánh giá" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "Mô tả" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "Nhận diện" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "Kiểu nhận diện" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "Giá trị nhận diện" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "Xoá" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "Thêm nhận diện" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "Đánh dấu" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "Series ID" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "Đánh giá" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "Lấy ảnh bìa từ URL (ảnh JPEG sẽ được tải xuống và lưu trong cơ sở dữ liệu)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "Tải ảnh bìa từ ổ cứng" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "Ngày xuất bản" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "Nhà xuất bản" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "Ngôn ngữ" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "Có" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "Không" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "Định dạng tải lên" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "Xem sách sau khi lưu" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "Fetch Metadata" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2003,37 +2077,31 @@ msgstr "Fetch Metadata" msgid "Save" msgstr "Lưu" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "Từ khoá" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "Tìm kiếm từ khoá" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "Ấn vào ảnh cover để tải metadata vào form" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "Đang tải..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "Đóng" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "Nguồn" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "Tìm kiếm lỗi!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "Không tìm thấy kết quả! Hãy thử lại với từ khoá khác." @@ -2140,7 +2208,7 @@ msgid "Enter " msgstr "Nhập " #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "Bạn có thực sự chắc chắn?" @@ -2614,74 +2682,61 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "Thêm cho phép/từ chối giá trị cột tuỳ biến" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "Đọc trên trình duyệt" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "Nghe trên trình duyệt" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "Sách %(index)s của %(range)s" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "Đã phát hành" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "Đánh dấu chưa đọc" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "Đánh dấu đã đọc" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "Đánh dấu chưa đọc" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "Đọc" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "Khôi phục từ lưu trữ" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "Thêm vào lưu trữ" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "Lưu trữ" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "Mô tả:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "Thêm vào giá sách" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(Công khai)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "Chỉnh sửa Metadata" @@ -2754,10 +2809,6 @@ msgstr "Nhập tên domain" msgid "Denied Domains (Blacklist)" msgstr "Domain bị từ chối (Blacklist)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "Tiếp tục" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2779,11 +2830,16 @@ msgstr "" msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "Thiết lập cơ sở dữ lieu :)))" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "Quay về trang chủ" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2873,7 +2929,7 @@ msgstr "Sách sắp xếp theo xếp hạng" msgid "Books ordered by file formats" msgstr "Sách sắp xếp theo định dạng" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "Giá sách" @@ -2882,60 +2938,36 @@ msgstr "Giá sách" msgid "Books organized in shelves" msgstr "Sách tổ chức theo giá sách" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "Trang chủ" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "Tìm kiếm thư viện" +#: cps/templates/layout.html:59 +msgid "Simple Theme" +msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "Tài khoản" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "Đăng xuất" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "Đang tải lên..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "Lỗi" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "Tải lên xong, đang xử lý, đợi chút ..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "Thiết lập" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "Làm ơn đừng load lại trang" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "Tìm kiếm" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "About" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "Trước" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "Chi tiết sách" @@ -3051,7 +3083,7 @@ msgstr "Thư mục cha" msgid "Select" msgstr "Chọn" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "Ok" @@ -3059,35 +3091,81 @@ msgstr "Ok" msgid "Calibre-Web eBook Catalog" msgstr "" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "trình đọc epub" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "Chọn tên người dùng" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "Sáng" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "Tối" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "Quay lại" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "Xoá" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "Đang chờ" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "Dọc" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "Đọc" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "Cột đọc không hợp lệ" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Trình đọc truyện tranh" @@ -3265,10 +3343,6 @@ msgstr "Link xác nhận này sẽ hết hạn trong 10 phút." msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "Không tìm thấy kết quả" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "Từ khoá tìm kiếm:" @@ -3285,11 +3359,11 @@ msgstr "Ngày phát hành từ" msgid "Published Date To" msgstr "Ngày phát hành tới" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3325,11 +3399,13 @@ msgstr "Đánh giá cao hơn" msgid "Rating Below" msgstr "Đánh giá thấp hơn" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "Từ:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "Tới:" @@ -3353,6 +3429,14 @@ msgstr "Khoá thay đổi thứ tự" msgid "Enable Change order" msgstr "Mở thay đổi thứ tự" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "Chia sửa với tất cả" @@ -3421,16 +3505,21 @@ msgstr "Tiến độ" msgid "Run Time" msgstr "Thời gian chạy" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "Trộn" + +#: cps/templates/tasks.html:21 #, fuzzy msgid "Actions" msgstr "Đánh giá" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3438,6 +3527,10 @@ msgstr "" msgid "Reset user Password" msgstr "Reset mật khẩu người dùng" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "Ngôn ngữ của sách" diff --git a/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.mo b/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.mo index c260b191..b11563e1 100644 Binary files a/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.mo and b/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.po b/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.po index 3637398b..f83637a2 100644 --- a/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.po +++ b/cps/translations/zh_Hans_CN/LC_MESSAGES/messages.po @@ -7,506 +7,524 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" -"PO-Revision-Date: 2020-09-27 22:18+0800\n" -"Last-Translator: xlivevil \n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" +"PO-Revision-Date: 2024-11-21 22:04+0800\n" +"Last-Translator: qx100\n" "Language: zh_CN\n" "Language-Team: zh_Hans_CN \n" "Plural-Forms: nplurals=1; plural=0;\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "统计" -#: cps/admin.py:150 +#: cps/admin.py:151 msgid "Server restarted, please reload page." -msgstr "服务器已重启,请刷新页面" +msgstr "服务器已重启,请刷新页面。" -#: cps/admin.py:152 +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." -msgstr "正在关闭服务器,请关闭窗口" +msgstr "正在关闭服务器,请关闭窗口。" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "数据库重新连接成功" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "未知命令" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "成功!书籍已排队进行元数据备份,请检查任务列表以获取结果" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "未知" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "管理页" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "基本配置" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "界面配置" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "自定义列号:%(column)d 在 Calibre 数据库中不存在" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "管理用户" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "全部" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "找不到用户" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "成功删除 {} 个用户" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "显示全部" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "格式错误的请求" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "访客名称无法更改" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" -msgstr "游客无法拥有此角色" +msgstr "访客无法拥有此角色" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "没有其余管理员账户,无法删除管理员角色" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "值必须为 true 或 false" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "无效角色" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" -msgstr "游客无法查看此页面" +msgstr "访客无法查看此页面" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "无效页面" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" -msgstr "无法设置游客的本地化,该项设置的值将自动检测" +msgstr "无法设置访客的本地化,该项设置的值将自动检测" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "无可用本地化" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "无有效书籍语言" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "参数未找到" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "无效的阅读栏目" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "无效的限制栏目" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web 配置已更新" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "您确定删除 Kobo 令牌吗?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "您确定要删除此域吗?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "您确定要删除此用户吗?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "您确定要删除此书架吗?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "您确定要修改选定用户的本地化设置吗?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "您确定要修改选定用户的可见书籍语言吗?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "您确定要修改选定用户的选定角色吗?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "您确定要修改选定用户的选定限制吗?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "您确定要修改选定用户的选定可视化限制吗?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "您确定要更改所选用户的书架同步行为吗?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "您确定要更改 Calibre 库位置吗?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" -msgstr "Calibre-Web 将搜索更新封面,并更新缩略图,这可能需要一段时间" +msgstr "Calibre-Web 将搜索更新封面,并更新缩略图,这可能需要一段时间?" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" -msgstr "您确定要删除 Calibre-Web 的同步数据库以强制与您的 Kobo Reader 进行完全同步吗" +msgstr "您确定要删除 Calibre-Web 的同步数据库以强制与您的 Kobo Reader 进行完全同步吗?" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "拒绝" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "允许" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "{} 同步项目被删除" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "标签未找到" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "无效的动作" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json 未为 Web 应用程序配置" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "日志文件路径无效,请输入正确的路径" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "访问日志路径无效,请输入正确的路径" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "请输入 LDAP 主机、端口、DN 和用户对象标识符" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" -msgstr "请输入一个 LDAP 服务账号和密码 " +msgstr "请输入一个 LDAP 服务账号和密码" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "请输入一个 LDAP 服务账号" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP 组对象过滤器需要一个具有“%s”格式标识符" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAP 组对象过滤器的括号不匹配" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP 用户对象过滤器需要一个具有“%s”格式标识符" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAP 用户对象过滤器的括号不匹配" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP 成员用户过滤器需要有一个“%s”格式标识符" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "LDAP 成员用户过滤器中有不匹配的括号" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CA证书、证书或密钥位置无效,请输入正确的路径" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "添加新用户" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "编辑邮件服务器设置" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." -msgstr "Gmail 账户验证成功" +msgstr "Gmail 账户验证成功。" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." -msgstr "数据库错误:%(error)s" +msgstr "数据库错误:%(error)s。" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "发送给 %(email)s 的测试邮件已加入队列。请检查任务结果" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "发送测试邮件时出错:%(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "请先配置您的邮箱地址..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "邮件服务器设置已更新" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "编辑计划任务设置" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "指定任务的开始时间无效" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "指定任务的持续时间无效" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "已更新计划任务设置" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." -msgstr "发生一个未知错误,请稍后再试" +msgstr "发生一个未知错误,请稍后再试。" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "设置数据库不可写" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "编辑用户 %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "用户 %(user)s 的密码已重置" -#: cps/admin.py:1451 +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." msgstr "请先配置 SMTP 邮箱设置..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "日志文件查看器" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "正在请求更新包" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "正在下载更新包" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "正在解压更新包" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "正在替换文件" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "数据库连接已关闭" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "正在停止服务器" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "更新完成,请点击确定并刷新页面" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "更新失败:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP 错误" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "连接错误" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "建立连接超时" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "一般错误" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "更新文件无法保存在临时目录中" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "更新期间无法替换文件" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "未能获得任何 LDAP 用户" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "未能创建任何 LDAP 用户" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "错误:%(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "错误:在 LDAP 服务器的响应中没有返回用户" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "数据库中没有找到任何 LDAP 用户" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} 用户被成功导入" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "书籍路径无效" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "数据库路径无效,请输入正确的路径" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "数据库不可写入" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "密钥文件路径无效,请输入正确的路径" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "证书文件路径无效,请输入正确的路径" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "密码长度必须在1到40之间" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "数据库设置已更新" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "数据库配置" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." -msgstr "请填写所有字段!" +msgstr "请填写所有字段。" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" -msgstr "邮箱不在有效域中" +msgstr "邮箱域名无效" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "添加新用户" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "用户“%(user)s”已创建" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." -msgstr "使用此邮箱或用户名的账号已经存在" +msgstr "使用此邮箱或用户名的账号已经存在。" -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "用户“%(nick)s”已删除" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" -msgstr "无法删除游客用户" +msgstr "无法删除访客用户" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "管理员账户不存在,无法删除用户" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "电子邮件地址不能为空,并且必须是有效的电子邮件" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "用户“%(nick)s”已更新" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "搜索" + #: cps/converter.py:31 msgid "not installed" msgstr "未安装" @@ -515,128 +533,122 @@ msgstr "未安装" msgid "Execution permissions missing" msgstr "缺少执行权限" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "自定义列号:%(column)d 在 Calibre 数据库中不存在" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "无" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "糟糕!选择书名无法打开。文件不存在或者文件不可访问" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "用户没有权限上传封面" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "标识符不区分大小写,覆盖旧标识符" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "已成功更新元数据" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "编辑书籍时出错: {}" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "文件 %(file)s 已上传" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "转换的源格式或目的格式缺失" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "书籍已经被成功加入 %(book_format)s 格式转换队列" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "转换此书籍时出现错误: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "上传的书籍可能已经存在,建议修改后重新上传: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "糟糕!选择书名无法打开。文件不存在或者文件不可访问" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "用户没有权限上传封面" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "标识符不区分大小写,覆盖旧标识符" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "'%(langname)s' 不是一种有效语言" -#: cps/editbooks.py:756 cps/editbooks.py:1192 -#, fuzzy -msgid "File type isn't allowed to be uploaded to this server" -msgstr "不能上传文件扩展名为“%(ext)s”的文件到此服务器" +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "已成功更新元数据" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "编辑书籍时出错: {}" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "上传的书籍可能已经存在,建议修改后重新上传: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 +msgid "File type isn't allowed to be uploaded to this server" +msgstr "此文件类型不允许上传到服务器" + +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "不能上传文件扩展名为“%(ext)s”的文件到此服务器" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "要上传的文件必须具有扩展名" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "文件 %(filename)s 无法保存到临时目录" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "移动封面文件失败 %(file)s:%(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "书籍的此格式副本已成功删除" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "书籍已成功删除" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "您没有删除书籍的权限" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "编辑元数据" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" -msgstr "%(seriesindex)s 不是一个有效的数值,忽略" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" +msgstr "丛书索引: %(seriesindex)s 不是一个有效的数值,跳过" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "用户没有权限上传其他文件格式" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "创建路径 %(path)s 失败 (权限不足)" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." -msgstr "保存文件 %(file)s 失败" +msgstr "保存文件 %(file)s 失败。" -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "已添加 %(ext)s 格式到 %(book)s" @@ -649,470 +661,461 @@ msgstr "Google Drive 设置未完成,请尝试停用并再次激活 Google 云 msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "回调域名尚未被校验,请在 Google 开发者控制台按步骤校验域名" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "找不到 ID 为 %(book)d 的书籍的 %(format)s 格式" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "Google Drive %(fn)s 上找不到 %(format)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "找不到 %(format)s:%(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "发送到电子阅读器" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." -msgstr "此邮件已经通过 Calibre-Web 发送" +msgstr "此邮件已经通过 Calibre-Web 发送。" -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "Calibre-Web 测试邮件" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "测试邮件" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "开启 Calibre-Web 之旅" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "用户注册电子邮件:%(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "转换 %(orig)s 到 %(format)s 并发送到电子阅读器" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, python-format msgid "Send %(format)s to eReader" msgstr "发送 %(format)s 到电子阅读器" -#: cps/helper.py:227 +#: cps/helper.py:230 #, python-format msgid "%(book)s send to eReader" msgstr "%(book)s 发送到电子阅读器" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "无法读取请求的文件。可能有错误的权限设置?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "阅读状态无法设置: {}" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "删除书的文件夹 %(id)s 失败,路径有子文件夹:%(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "删除书籍 %(id)s 失败:%(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "仅从数据库中删除书籍 %(id)s,数据库中的书籍路径无效: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "将作者从“%(src)s”改为“%(dest)s”时失败,出错信息:%(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Google Drive 上找不到文件 %(file)s" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "将标题从“%(src)s”改为“%(dest)s”时失败,出错信息:%(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Google Drive 上找不到书籍路径 %(path)s" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "已存在使用此邮箱的账户" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "此用户名已被使用" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "无效的邮箱格式" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "密码不符合密码验证规则" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "上传封面所需的 Python 模块 'advocate' 未安装" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "下载封面时出错" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "封面格式出错" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "您没有访问本地主机或本地网络进行封面上传" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "创建封面路径失败" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "封面文件不是有效的图片文件,或者无法存储它" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "封面文件只支持 jpg、jpeg、png、webp、bmp 文件" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "封面文件内容无效" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "仅将 jpg、jpeg 文件作为封面文件" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" msgstr "封面" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "找不到 UnRar 执行文件" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "执行 UnRar 时出错" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" -msgstr "" +msgstr "无法找到指定目录" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" -msgstr "" +msgstr "请指定目录,而不是文件" -#: cps/helper.py:1064 -#, fuzzy +#: cps/helper.py:1042 msgid "Calibre binaries not viable" -msgstr "数据库不可写入" +msgstr "Calibre 程序无法运行" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" -msgstr "" +msgstr "缺失的 calibre 程序:%(missing)s" -#: cps/helper.py:1075 -#, fuzzy, python-format +#: cps/helper.py:1053 +#, python-format msgid "Missing executable permissions: %(missing)s" -msgstr "缺少执行权限" +msgstr "缺少执行权限: %(missing)s" -#: cps/helper.py:1080 -#, fuzzy +#: cps/helper.py:1058 msgid "Error executing Calibre" -msgstr "执行 UnRar 时出错" +msgstr "执行 Calibre 时出错" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "将所有书籍加入元数据备份队列" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "请不要从 localhost 访问 Calibre-Web,以便 Kobo 设备能获取有效的 api_endpoint" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo 设置" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "使用 %(provider)s 注册" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "您现在已以“%(nickname)s”身份登录" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "链接到 %(oauth)s 成功" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "登录失败,没有用户与 OAuth 帐户关联" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "取消链接到 %(oauth)s 成功" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "取消链接到 %(oauth)s 失败" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "未连接到 %(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." -msgstr "使用 Github 登录失败" +msgstr "使用 Github 登录失败。" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." -msgstr "从 Github 获取用户信息失败" +msgstr "从 Github 获取用户信息失败。" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." -msgstr "使用 Google 登录失败" +msgstr "使用 Google 登录失败。" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." -msgstr "从 Google 获取用户信息失败" +msgstr "从 Google 获取用户信息失败。" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." -msgstr "GitHub Oauth 错误,请重试" +msgstr "GitHub Oauth 错误,请重试。" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "GitHub Oauth 错误: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." -msgstr "Google Oauth 错误,请重试" +msgstr "Google Oauth 错误,请重试。" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google Oauth 错误: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} 星" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "登录" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "找不到令牌" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "令牌已过期" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "成功!请返回您的设备" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "书籍" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "显示最近查看的书籍" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "热门书籍" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "显示热门书籍" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "已下载书籍" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "显示下载过的书籍" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "最高评分书籍" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "显示最高评分书籍" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "已读书籍" -#: cps/render_template.py:64 +#: cps/render_template.py:63 msgid "Show Read and Unread" msgstr "显示已读或未读状态" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "未读书籍" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "显示未读" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "发现" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "显示随机书籍" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "分类" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" msgstr "显示分类栏目" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "丛书" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" msgstr "显示丛书栏目" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "作者" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" msgstr "显示作者栏目" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "出版社" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" msgstr "显示出版社栏目" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "语言" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" msgstr "显示语言栏目" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "评分" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" msgstr "显示评分栏目" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "文件格式" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" msgstr "显示文件格式栏目" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "归档书籍" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "显示归档书籍" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "书籍列表" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "显示书籍列表" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "搜索" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "出版时间晚于 " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "出版时间早于 " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "评分 <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "评分 >= %(rating)s" -#: cps/search.py:221 -#, fuzzy, python-format +#: cps/search.py:234 +#, python-format msgid "Read Status = '%(status)s'" -msgstr "阅读状态 = %(status)s" +msgstr "阅读状态 = '%(status)s'" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "搜索自定义栏目时出错,请重启 Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "高级搜索" @@ -1132,7 +1135,7 @@ msgstr "此书籍已经是书架 %(shelfname)s 的一部分" #: cps/shelf.py:77 #, python-format msgid "%(book_id)s is a invalid Book Id. Could not be added to Shelf" -msgstr "" +msgstr "%(book_id)s 无效 无法添加到书架" #: cps/shelf.py:97 #, python-format @@ -1167,7 +1170,7 @@ msgstr "此书已从书架 %(sname)s 中删除" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "抱歉,您没有从这个书架删除书籍的权限" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "创建书架" @@ -1213,52 +1216,52 @@ msgstr "发生错误" #: cps/shelf.py:380 #, python-format msgid "A public shelf with the name '%(title)s' already exists." -msgstr "公共书架:%(title)s 已经存在已经存在" +msgstr "公共书架:%(title)s 已经存在已经存在。" #: cps/shelf.py:391 #, python-format msgid "A private shelf with the name '%(title)s' already exists." -msgstr "私有书架:%(title)s 已经存在已经存在" +msgstr "私有书架:%(title)s 已经存在已经存在。" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "书架:%(name)s" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "打开书架出错。书架不存在或不可访问" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "任务列表" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "等待中" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "失败" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "已开始" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "已完成" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "已结束" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "已取消" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "未知状态" @@ -1272,7 +1275,7 @@ msgstr "无可用更新。您已经安装了最新版本" #: cps/updater.py:458 msgid "A new update is available. Click on the button below to update to the latest version." -msgstr "有新的更新。单击下面的按钮以更新到最新版本" +msgstr "有新的更新。单击下面的按钮以更新到最新版本。" #: cps/updater.py:476 msgid "Could not fetch update information" @@ -1280,7 +1283,7 @@ msgstr "无法获取更新信息" #: cps/updater.py:486 msgid "Click on the button below to update to the latest stable version." -msgstr "点击下面按钮更新到最新稳定版本" +msgstr "点击下面按钮更新到最新稳定版本。" #: cps/updater.py:495 cps/updater.py:509 cps/updater.py:520 #, python-format @@ -1291,225 +1294,228 @@ msgstr "有新的更新。单击下面的按钮以更新到版本: %(version)s" msgid "No release information available" msgstr "无可用发布信息" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "发现 (随机书籍)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "热门书籍(最多下载)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "%(user)s 下载过的书籍" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "作者:%(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "出版社:%(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "丛书:%(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "评分:无" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "评分:%(rating)s 星" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "文件格式:%(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "分类:%(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "语言:%(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "下载次数" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "评分列表" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "文件格式列表" -#: cps/web.py:1249 +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." msgstr "请先配置 SMTP 邮箱设置..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "书籍已经成功加入 %(eReadermail)s 的发送队列" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "糟糕!发送这本书籍的时候出现错误:%(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." -msgstr "请先配置您的 Kindle 邮箱" +msgstr "请先配置您的 Kindle 邮箱。" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "请等待一分钟注册下一个用户" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "注册" -#: cps/web.py:1281 cps/web.py:1385 -#, fuzzy +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" -msgstr "邮件服务未配置,请联系网站管理员!" +msgstr "限制器后台连接错误,请联系管理员" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." -msgstr "邮件服务未配置,请联系网站管理员!" +msgstr "邮件服务未配置,请联系网站管理员。" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." -msgstr "您的电子邮件不允许注册" +msgstr "您的电子邮件不允许注册。" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." -msgstr "确认邮件已经发送到您的邮箱" +msgstr "确认邮件已经发送到您的邮箱。" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "无法激活 LDAP 认证" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "下次登录前请等待一分钟" -#: cps/web.py:1400 +#: cps/web.py:1408 #, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "您现在已以“%(nickname)s”身份登录" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "后备登录“%(nickname)s”:无法访问 LDAP 服务器,或用户未知" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "无法登录:%(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" msgstr "用户名或密码错误" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "新密码已发送到您的邮箱" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." -msgstr "发生一个未知错误,请稍后再试" +msgstr "发生一个未知错误,请稍后再试。" -#: cps/web.py:1429 +#: cps/web.py:1437 msgid "Please enter valid username to reset password" msgstr "请输入有效的用户名进行密码重置" -#: cps/web.py:1437 +#: cps/web.py:1445 #, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "您现在已以“%(nickname)s”身份登录" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s 的用户配置" -#: cps/web.py:1511 +#: cps/web.py:1526 msgid "Success! Profile Updated" msgstr "资料已更新" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." -msgstr "使用此邮箱的账号已经存在" +msgstr "使用此邮箱的账号已经存在。" #: cps/services/gmail.py:59 msgid "Found no valid gmail.json file with OAuth information" msgstr "找不到包含 OAuth 信息的有效 gmail.json 文件" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "删除临时文件夹" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s 发送到电子阅读器" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "Calibre 电子书转换器 %(tool)s 没有发现" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "硬盘上找不到 %(format)s 格式" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "发生未知错误,书籍转换失败" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify 转换失败:%(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "找不到转换后的文件或文件夹 %(folder)s 中有多个文件" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "电子书转换器失败: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre 运行失败,错误信息:%(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "电子书转换器失败: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "转换" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "正在重新连接到 Calibre 数据库" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "电子邮件" @@ -1517,30 +1523,26 @@ msgstr "电子邮件" msgid "Backing up Metadata" msgstr "正在备份元数据" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "生成了 %(count)s 个封面缩略图" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "封面缩略图" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" -msgstr "生成了 %(count)s 个丛书缩略图" +msgstr "生成了 {0} 个丛书缩略图" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "正在清理封面缩略图缓存" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "上传书籍" @@ -1559,11 +1561,11 @@ msgstr "用户名" msgid "Email" msgstr "邮箱地址" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "接收书籍的电子阅读器邮箱地址" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "管理权限" @@ -1573,8 +1575,8 @@ msgstr "管理权限" msgid "Password" msgstr "密码" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "下载书籍" @@ -1784,13 +1786,13 @@ msgid "OK" msgstr "确定" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "取消" @@ -1813,12 +1815,12 @@ msgstr "在书库" #: cps/templates/author.html:26 cps/templates/index.html:74 #: cps/templates/search.html:31 cps/templates/shelf.html:20 msgid "Sort according to book date, newest first" -msgstr "按图书日期排序,最新优先" +msgstr "按书籍日期排序,最新优先" #: cps/templates/author.html:27 cps/templates/index.html:75 #: cps/templates/search.html:32 cps/templates/shelf.html:21 msgid "Sort according to book date, oldest first" -msgstr "按图书日期排序,最旧优先" +msgstr "按书籍日期排序,最旧优先" #: cps/templates/author.html:28 cps/templates/index.html:76 #: cps/templates/search.html:33 cps/templates/shelf.html:22 @@ -1840,16 +1842,76 @@ msgstr "按出版日期排序,最新优先" msgid "Sort according to publishing date, oldest first" msgstr "按出版日期排序,最旧优先" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "减少" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "更多" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "%(range)s 第 %(index)s 册" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "语言" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "出版社" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "出版日期" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "简介:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "上一个" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "下一个" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "无搜索结果" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "首页" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "搜索书库" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "注销" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "删除书籍" @@ -1878,99 +1940,107 @@ msgstr "转换到:" msgid "Convert book" msgstr "转换书籍" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "正在上传..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "关闭" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "错误" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "上传完成,正在处理,请稍候..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "上传格式" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "书名" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "作者" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "标签" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "丛书编号" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "出版日期" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "评分" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "简介" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "书号" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "书号类型" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "书号编号" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "移除" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "添加书号" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "标签" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "丛书编号" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "评分" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "从 URL 获取封面(JPEG - 图片将下载并存储在数据库中)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "从本地磁盘上传封面" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "出版日期" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "出版社" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "语言" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "确认" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "没有" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "上传格式" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "查看保存书籍" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "获取元数据" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -1978,39 +2048,33 @@ msgstr "获取元数据" msgid "Save" msgstr "保存" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "关键字" -#: cps/templates/book_edit.html:234 -msgid "Search keyword" -msgstr " 搜索关键字 " - #: cps/templates/book_edit.html:240 +msgid "Search keyword" +msgstr "搜索关键字" + +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "单击封面将元数据加载到表单" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "加载中..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "关闭" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "源" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "搜索错误!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." -msgstr "无搜索结果!请尝试另一个关键字" +msgstr "无搜索结果!请尝试另一个关键字。" #: cps/templates/book_table.html:12 cps/templates/book_table.html:69 #: cps/templates/user_table.html:14 cps/templates/user_table.html:77 @@ -2115,7 +2179,7 @@ msgid "Enter " msgstr "输入" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "您真的确认?" @@ -2137,7 +2201,7 @@ msgstr "Calibre 数据库路径" #: cps/templates/config_db.html:21 msgid "Separate Book Files from Library" -msgstr "" +msgstr "将书籍文件与书库分开" #: cps/templates/config_db.html:34 msgid "Use Google Drive?" @@ -2221,7 +2285,7 @@ msgstr "保存到磁盘时转换标题和作者中的非英文字符" #: cps/templates/config_edit.html:108 msgid "Embed Metadata to Ebook File on Download/Conversion/e-mail (needs Calibre/Kepubify binaries)" -msgstr "" +msgstr "在下载/转换/电子邮件时将元数据嵌入电子书文件(需要 Calibre/Kepubify 程序)" #: cps/templates/config_edit.html:112 msgid "Enable Uploads" @@ -2411,9 +2475,8 @@ msgid "External binaries" msgstr "扩展程序配置" #: cps/templates/config_edit.html:325 -#, fuzzy msgid "Path to Calibre Binaries" -msgstr "Calibre 电子书转换器路径" +msgstr "Calibre 程序的路径" #: cps/templates/config_edit.html:333 msgid "Calibre E-Book Converter Settings" @@ -2424,9 +2487,8 @@ msgid "Path to Kepubify E-Book Converter" msgstr "KEpubify 电子书转换器路径" #: cps/templates/config_edit.html:344 -#, fuzzy msgid "Location of Unrar binary" -msgstr "UnRar 程序路径" +msgstr "Unrar 程序的位置" #: cps/templates/config_edit.html:360 msgid "Security Settings" @@ -2438,15 +2500,15 @@ msgstr "限制失败的登录尝试" #: cps/templates/config_edit.html:372 msgid "Configure Backend for Limiter" -msgstr "" +msgstr "配置限制器后台" #: cps/templates/config_edit.html:376 msgid "Options for Limiter Backend" -msgstr "" +msgstr "限制器后台选项" #: cps/templates/config_edit.html:382 msgid "Check if file extensions matches file content on upload" -msgstr "" +msgstr "检查上传的文件扩展名是否与文件内容匹配" #: cps/templates/config_edit.html:385 msgid "Session protection" @@ -2482,7 +2544,7 @@ msgstr "必须使用大写字符" #: cps/templates/config_edit.html:414 msgid "Enforce characters (needed For Chinese/Japanese/Korean Characters)" -msgstr "" +msgstr "必须使用文字(需要中文/日文/韩文文字)" #: cps/templates/config_edit.html:418 msgid "Enforce special characters" @@ -2589,74 +2651,59 @@ msgstr "添加显示或隐藏书籍的标签值" msgid "Add Allowed/Denied custom column values" msgstr "添加显示或隐藏书籍的自定义栏目值" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "在线阅读" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "在线听书" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "%(range)s 第 %(index)s 册" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "出版日期" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "标为未读" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "标为已读" -#: cps/templates/detail.html:254 -#, fuzzy +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" -msgstr "标为未读" +msgstr "将书籍标记为已读或未读" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "已读" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "从档案还原" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "添加到归档" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" -msgstr "" +msgstr "将书籍标记为已存档或未存档,以便在 Calibre-Web 中隐藏书籍并从 Kobo 阅读器中删除书籍" -#: cps/templates/detail.html:267 -#, fuzzy +#: cps/templates/detail.html:275 msgid "Archive" msgstr "归档" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "简介:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "添加到书架" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(公共)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "编辑元数据" @@ -2726,10 +2773,6 @@ msgstr "输入域名" msgid "Denied Domains (Blacklist)" msgstr "禁止注册的域名(黑名单)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "下一个" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "在文本编辑器中打开 .kobo/Kobo/Kobo eReader.conf,添加(或编辑):" @@ -2744,17 +2787,22 @@ msgstr "列表" #: cps/templates/http_error.html:34 msgid "Calibre-Web Instance is unconfigured, please contact your administrator" -msgstr "Calibre-Web 实例未配置,请联系您的管理员!" +msgstr "Calibre-Web 实例未配置,请联系您的管理员" #: cps/templates/http_error.html:44 msgid "Create Issue" msgstr "创建问题" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "数据库配置" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "回到首页" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "登出账号" @@ -2798,11 +2846,11 @@ msgstr "按字母排序的书籍" #: cps/templates/index.xml:31 msgid "Popular publications from this catalog based on Downloads." -msgstr "基于下载数的热门书籍" +msgstr "基于下载数的热门书籍。" #: cps/templates/index.xml:40 msgid "Popular publications from this catalog based on Rating." -msgstr "基于评分的热门书籍" +msgstr "基于评分的热门书籍。" #: cps/templates/index.xml:45 msgid "Recently added Books" @@ -2844,7 +2892,7 @@ msgstr "书籍按评分排序" msgid "Books ordered by file formats" msgstr "书籍按文件格式排序" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "书架列表" @@ -2853,60 +2901,37 @@ msgstr "书架列表" msgid "Books organized in shelves" msgstr "书架上的书" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "首页" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "切换导航" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "搜索书库" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "简单" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "账号" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "注销" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "正在上传..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "错误" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "上传完成,正在处理,请稍候..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "设置" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "请不要刷新页面" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "按条件浏览" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "关于" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "上一个" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "书籍详情" @@ -2992,11 +3017,11 @@ msgstr ",包括从硬盘中" #: cps/templates/modal_dialogs.html:56 msgid "Important Kobo Note: deleted books will remain on any paired Kobo device." -msgstr "Kobo 重要提示:被删除的书籍将保留在任何配对的 Kobo 设备上" +msgstr "Kobo 重要提示:被删除的书籍将保留在任何配对的 Kobo 设备上。" #: cps/templates/modal_dialogs.html:57 msgid "Books must first be archived and the device synced before a book can safely be deleted." -msgstr "在安全删除图书之前,必须先将图书归档并与设备同步" +msgstr "在安全删除书籍之前,必须先将书籍归档并与设备同步。" #: cps/templates/modal_dialogs.html:76 msgid "Choose File Location" @@ -3022,7 +3047,7 @@ msgstr "父目录" msgid "Select" msgstr "选择" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "完成" @@ -3030,34 +3055,75 @@ msgstr "完成" msgid "Calibre-Web eBook Catalog" msgstr "Caliebre-Web 电子书路径" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "epub 阅读器" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "选择一个用户名" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "浅色" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "深色" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "棕色" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "黑色" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." -msgstr "打开侧栏时重排文本" +msgstr "打开侧栏时重排文本。" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "字体大小" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "字体" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "默认" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "微软雅黑" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "宋体" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "楷体" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "Arial" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "分配" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "双栏" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "单栏" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Comic 阅读器" @@ -3172,11 +3238,11 @@ msgstr "从右到左" #: cps/templates/readcbr.html:162 msgid "Reset to Top" -msgstr "" +msgstr "返回页首" #: cps/templates/readcbr.html:163 msgid "Remember Position" -msgstr "" +msgstr "记住位置" #: cps/templates/readcbr.html:168 msgid "Scrollbar" @@ -3224,20 +3290,16 @@ msgstr "在另一个设备上,登录并访问:" #: cps/templates/remote_login.html:11 msgid "Once verified, you will automatically be logged in on this device." -msgstr "验证后,您将自动在新设备上登录" +msgstr "验证后,您将自动在新设备上登录。" #: cps/templates/remote_login.html:14 msgid "This verification link will expire in 10 minutes." -msgstr "此验证链接将在10分钟后失效" +msgstr "此验证链接将在10分钟后失效。" #: cps/templates/schedule_edit.html:33 msgid "Generate Series Cover Thumbnails" msgstr "生成丛书封面缩略图" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "无搜索结果" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "搜索项:" @@ -3254,13 +3316,13 @@ msgstr "出版日期从" msgid "Published Date To" msgstr "出版日期到" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" -msgstr "" +msgstr "任何" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" -msgstr "" +msgstr "空" #: cps/templates/search_form.html:60 msgid "Exclude Tags" @@ -3294,11 +3356,13 @@ msgstr "评分大于" msgid "Rating Below" msgstr "评分小于" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "从:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "到:" @@ -3322,6 +3386,14 @@ msgstr "禁止改变顺序" msgid "Enable Change order" msgstr "允许改变顺序" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "按添加日期排序,最新优先" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "按添加日期排序,最旧优先" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "书架将被公开" @@ -3390,22 +3462,30 @@ msgstr "任务进度" msgid "Run Time" msgstr "运行时间" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "消息" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "活动" -#: cps/templates/tasks.html:40 -msgid "This task will be cancelled. Any progress made by this task will be saved." -msgstr "这个任务将被取消。此任务所有的更改都将被保存" - #: cps/templates/tasks.html:41 +msgid "This task will be cancelled. Any progress made by this task will be saved." +msgstr "这个任务将被取消。此任务所有的更改都将被保存。" + +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." -msgstr "如果这是计划任务,则将在下一个计划的时间内重新运行" +msgstr "如果这是计划任务,则将在下一个计划的时间内重新运行。" #: cps/templates/user_edit.html:20 msgid "Reset user Password" msgstr "重置用户密码" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "发送至电子阅读器的电子邮箱地址。使用逗号分隔多个邮箱地址" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "按语言显示书籍" diff --git a/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.mo b/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.mo index 58fa6319..24ff5edb 100644 Binary files a/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.mo and b/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.mo differ diff --git a/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.po b/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.po index 15b25b65..2511717d 100644 --- a/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.po +++ b/cps/translations/zh_Hant_TW/LC_MESSAGES/messages.po @@ -7,7 +7,7 @@ msgid "" msgstr "" "Project-Id-Version: Calibre-Web\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: 2020-09-27 22:18+0800\n" "Last-Translator: xlivevil \n" "Language: zh_TW\n" @@ -16,503 +16,521 @@ msgstr "" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "統計" -#: cps/admin.py:150 +#: cps/admin.py:151 #, fuzzy msgid "Server restarted, please reload page." msgstr "服務器已重啟,請刷新頁面" -#: cps/admin.py:152 +#: cps/admin.py:153 #, fuzzy msgid "Performing Server shutdown, please close window." msgstr "正在關閉服務器,請關閉窗口" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "未知命令" -#: cps/admin.py:174 +#: cps/admin.py:175 #, fuzzy msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "發送給%(email)s的測試郵件已進入隊列。請檢查任務結果" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "未知" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "管理頁" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "基本配置" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "界面配置" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, fuzzy, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "自定義列號:%(column)d在Calibre數據庫中不存在" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "管理用戶" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "全部" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "找不到用戶" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "成功刪除 {} 個用戶" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "顯示全部" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "格式錯誤的請求" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "訪客名稱無法更改" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "遊客無法擁有此角色" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "管理員賬戶不存在,無法刪除管理員角色" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "值必須是 true 或 false" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "無效角色" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "遊客無法擁有此視圖" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "無效視圖" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "訪客的本地化是自動偵測而無法設置的" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "無可用本地化" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "無有效書籍語言" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "參數未找到" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "無效的閱讀列" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "無效的限制列" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "Calibre-Web配置已更新" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "您確定刪除Kobo Token嗎?" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "您確定要刪除此網域嗎?" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "您確定要刪除此用戶嗎?" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "您確定要刪除此書架嗎?" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "您確定要修改選定用戶的本地化設置嗎?" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "您確定要修改選定用戶的可見書籍語言嗎?" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "您確定要修改選定用戶的選定角色嗎?" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "您確定要修改選定用戶的選定限制嗎?" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "您確定要修改選定用戶的選定可視化限制嗎?" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "您確定要更改所選用戶的書架同步行為嗎?" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "您確定要更改 Calibre 庫位置嗎?" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "拒絕" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "允許" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "標籤未找到" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "無效的動作" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "client_secrets.json 未為 Web 應用程序配置" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "日誌文件路徑無效,請輸入正確的路徑" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "訪問日誌路徑無效,請輸入正確的路徑" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "請輸入LDAP主機、端口、DN和用戶對象標識符" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "請輸入一個LDAP服務賬號和密碼 " -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "請輸入一個LDAP服務賬號" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "LDAP群組對象過濾器需要一個具有“%s”格式標識符號" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "LDAP群組對象過濾器的括號不匹配" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP用戶對象過濾器需要一個具有“%s”格式標識符" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "LDAP用戶對象過濾器的括號不匹配" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "LDAP成員用戶過濾器需要有一個“%s”格式標識符號" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "LDAP成員用戶過濾器中有不匹配的括號" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "LDAP CA證書、證書或密鑰位置無效,請輸入正確的路徑" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "添加新用戶" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "編輯郵件服務器設置" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "數據庫錯誤:%(error)s。" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "發送給%(email)s的測試郵件已進入隊列。請檢查任務結果" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "發送測試郵件時出錯:%(res)s" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "請先配置您的郵箱地址..." -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "郵件服務器設置已更新" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "發生一個未知錯誤,請稍後再試。" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "編輯用戶 %(nick)s" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, fuzzy, python-format msgid "Success! Password for user %(user)s reset" msgstr "用戶 %(user)s 的密碼已重置" -#: cps/admin.py:1451 +#: cps/admin.py:1463 #, fuzzy msgid "Oops! Please configure the SMTP mail settings." msgstr "請先配置SMTP郵箱設置..." -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "日誌文件查看器" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "正在請求更新包" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "正在下載更新包" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "正在解壓更新包" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "正在替換文件" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "數據庫連接已關閉" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "正在停止服務器" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "更新完成,請點擊確定並刷新頁面" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "更新失敗:" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "HTTP錯誤" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "連接錯誤" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "建立連接超時" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "一般錯誤" -#: cps/admin.py:1539 +#: cps/admin.py:1551 #, fuzzy msgid "Update file could not be saved in temp dir" msgstr "更新文件無法保存在臨時目錄中" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "更新時檔案無法替換變更" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "未能提取至少一個LDAP用戶" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "未能創建至少一個LDAP用戶" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "錯誤:%(ldaperror)s" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "錯誤:在LDAP服務器的響應中沒有返回用戶" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "數據庫中沒有找到至少一個LDAP用戶" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "{} 用戶被成功導入" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "數據庫路徑無效,請輸入正確的路徑" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "數據庫不可寫入" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "密鑰文件路徑無效,請輸入正確的路徑" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "證書文件路徑無效,請輸入正確的路徑" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 #, fuzzy msgid "Database Settings updated" msgstr "郵件服務器設置已更新" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "數據庫配置" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "請填寫所有欄位!" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "郵箱不在有效網域中" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "添加新用戶" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "用戶“%(user)s”已創建" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "使用此郵箱或用戶名的賬號已經存在。" -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "用戶“%(nick)s”已刪除" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "無法刪除訪客用戶" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "管理員賬戶不存在,無法刪除用戶" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "用戶“%(nick)s”已更新" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "搜尋" + #: cps/converter.py:31 msgid "not installed" msgstr "未安裝" @@ -521,128 +539,123 @@ msgstr "未安裝" msgid "Execution permissions missing" msgstr "缺少執行權限" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, fuzzy, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "自定義列號:%(column)d在Calibre數據庫中不存在" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "無" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "糟糕!選擇書名無法打開。文件不存在或者文件不可訪問" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "標識符不區分大小寫,覆蓋舊標識符" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "已成功更新元數據" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "文件 %(file)s 已上傳" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "轉換的來源或目的格式不存在" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "書籍已經被成功加入到 %(book_format)s 格式轉換隊列" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "轉換此書籍時出現錯誤: %(res)s" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " -msgstr "上傳的書籍可能已經存在,建議修改後重新上傳: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" +msgstr "糟糕!選擇書名無法打開。文件不存在或者文件不可訪問" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "標識符不區分大小寫,覆蓋舊標識符" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, fuzzy, python-format msgid "'%(langname)s' is not a valid language" msgstr "%(langname)s 不是一種有效語言" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "已成功更新元數據" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "上傳的書籍可能已經存在,建議修改後重新上傳: " + +#: cps/editbooks.py:755 cps/editbooks.py:1202 #, fuzzy msgid "File type isn't allowed to be uploaded to this server" msgstr "不能上傳文件附檔名為“%(ext)s”的文件到此服務器" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "不能上傳文件附檔名為“%(ext)s”的文件到此服務器" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "要上傳的文件必須具有附檔名" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "文件 %(filename)s 無法保存到臨時目錄" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "移動封面文件失敗 %(file)s:%(error)s" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "書籍格式已成功刪除" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "書籍已成功刪除" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "編輯元數據" -#: cps/editbooks.py:1013 -#, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +#: cps/editbooks.py:1016 +#, fuzzy, python-format +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "%(seriesindex)s 不是一個有效的數值,忽略" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "創建路徑 %(path)s 失敗(權限拒絕)。" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "保存文件 %(file)s 失敗。" -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "已添加 %(ext)s 格式到 %(book)s" @@ -655,487 +668,480 @@ msgstr "Google Drive 設置未完成,請嘗試停用並再次激活Google雲 msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "回調網域名稱尚未被驗證,請在google開發者控制台按步驟驗證網域名稱" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "找不到id為 %(book)d 的書籍的 %(format)s 格式" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "Google Drive %(fn)s 上找不到 %(format)s" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "找不到 %(format)s:%(fn)s" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 #, fuzzy msgid "Send to eReader" msgstr "發送到Kindle" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 #, fuzzy msgid "This Email has been sent via Calibre-Web." msgstr "此郵件已經通過Calibre-Web發送。" -#: cps/helper.py:120 +#: cps/helper.py:123 #, fuzzy msgid "Calibre-Web Test Email" msgstr "Calibre-Web測試郵件" -#: cps/helper.py:121 +#: cps/helper.py:124 #, fuzzy msgid "Test Email" msgstr "測試郵件" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "開啟Calibre-Web之旅" -#: cps/helper.py:143 +#: cps/helper.py:146 #, fuzzy, python-format msgid "Registration Email for user: %(name)s" msgstr "用戶註冊電子郵件:%(name)s" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, fuzzy, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "轉換 %(orig)s 到 %(format)s 並發送到Kindle" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, fuzzy, python-format msgid "Send %(format)s to eReader" msgstr "發送 %(format)s 到Kindle" -#: cps/helper.py:227 +#: cps/helper.py:230 #, fuzzy, python-format msgid "%(book)s send to eReader" msgstr "%(book)s發送到Kindle" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "無法讀取請求的文件。可能有錯誤的權限設置?" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "刪除書的文件夾%(id)s失敗,路徑有子文件夾:%(path)s" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "刪除書籍 %(id)s失敗:%(message)s" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "僅從數據庫中刪除書籍 %(id)s,數據庫中的書籍路徑無效: %(path)s" -#: cps/helper.py:463 +#: cps/helper.py:439 #, fuzzy, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "將標題從“%(src)s”改為“%(dest)s”時失敗,錯誤錯信息:%(error)s" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "Google Drive上找不到文件 %(file)s" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "將標題從“%(src)s”改為“%(dest)s”時失敗,錯誤錯信息:%(error)s" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "Google Drive上找不到書籍路徑 %(path)s" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "此用戶名已被使用" -#: cps/helper.py:702 +#: cps/helper.py:679 #, fuzzy msgid "Invalid Email address format" msgstr "無效的郵件地址格式" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "下載封面時出錯" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "封面格式出錯" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "創建封面路徑失敗" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "封面文件不是有效的圖片文件,或者無法儲存" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "封面文件只支持jpg/jpeg/png/webp/bmp格式文件" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "僅將jpg、jpeg文件作為封面文件" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 #, fuzzy msgid "Cover" msgstr "發現" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "找不到UnRar執行文件" -#: cps/helper.py:1039 +#: cps/helper.py:1017 #, fuzzy msgid "Error executing UnRar" msgstr "執行UnRar時出錯" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 #, fuzzy msgid "Calibre binaries not viable" msgstr "數據庫不可寫入" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, fuzzy, python-format msgid "Missing executable permissions: %(missing)s" msgstr "缺少執行權限" -#: cps/helper.py:1080 +#: cps/helper.py:1058 #, fuzzy msgid "Error executing Calibre" msgstr "執行UnRar時出錯" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 #, fuzzy msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "請不要使用localhost訪問Calibre-Web,以便Kobo設備能獲取有效的api_endpoint" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "Kobo 設置" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "使用 %(provider)s 註冊" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "您現在已以“%(nickname)s”身份登入" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "連接到%(oauth)s成功" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "登入失敗,沒有用戶與OAuth帳戶關聯" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "取消連接到%(oauth)s成功" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "取消連接到%(oauth)s失敗" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "為連接到%(oauth)s" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "使用Github登入失敗。" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "從Github獲取用戶信息失敗。" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "使用Google登入失敗。" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "從Google獲取用戶信息失敗。" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "GitHub Oauth 錯誤,請重試。" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "GitHub Oauth 錯誤: {}" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "Google Oauth 錯誤,請重試。" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "Google Oauth 錯誤: {}" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "{} 星" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "登入" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "找不到Token" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "Token已過期" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "成功!請返回您的設備" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "書籍" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "顯示最近書籍" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "熱門書籍" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "顯示熱門書籍" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "已下載書籍" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "顯示下載過的書籍" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "最高評分書籍" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "顯示最高評分書籍" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "已讀書籍" -#: cps/render_template.py:64 +#: cps/render_template.py:63 #, fuzzy msgid "Show Read and Unread" msgstr "顯示閱讀狀態" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "未讀書籍" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "顯示未讀" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "發現" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "隨機顯示書籍" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "分類" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 #, fuzzy msgid "Show Category Section" msgstr "顯示分類選擇" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "叢書" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 #, fuzzy msgid "Show Series Section" msgstr "顯示叢書選擇" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "作者" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 #, fuzzy msgid "Show Author Section" msgstr "顯示作者選擇" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "出版社" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 #, fuzzy msgid "Show Publisher Section" msgstr "顯示出版社選擇" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "語言" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 #, fuzzy msgid "Show Language Section" msgstr "顯示語言選擇" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "評分" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 #, fuzzy msgid "Show Ratings Section" msgstr "顯示評分選擇" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "文件格式" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 #, fuzzy msgid "Show File Formats Section" msgstr "顯示文件格式選擇" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "歸檔書籍" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 #, fuzzy msgid "Show Archived Books" msgstr "顯示歸檔書籍" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "書籍列表" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "顯示書籍列表" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "搜尋" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "出版時間晚於 " -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "出版時間早於 " -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "評分 <= %(rating)s" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "評分 >= %(rating)s" -#: cps/search.py:221 +#: cps/search.py:234 #, fuzzy, python-format msgid "Read Status = '%(status)s'" msgstr "閱讀狀態 = %(status)s" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "搜詢自定義欄位時出錯,請重啟 Calibre-Web" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "進階搜尋" @@ -1190,7 +1196,7 @@ msgstr "此書已從書架 %(sname)s 中刪除" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "創建書架" @@ -1244,45 +1250,45 @@ msgstr "公共書架:%(title)s已經存在已經存在。" msgid "A private shelf with the name '%(title)s' already exists." msgstr "私有書架:%(title)s已經存在。" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "書架:%(name)s" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "打開書架出錯。書架不存在或不可訪問" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "任務列表" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "等待中" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "失敗" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "已開始" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "已完成" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "未知狀態" @@ -1315,178 +1321,178 @@ msgstr "有新的更新。單擊下面的按鈕以更新到版本: %(version)s" msgid "No release information available" msgstr "無可用發佈信息" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "發現(隨機書籍)" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "熱門書籍(最多下載)" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "%(user)s 下載過的書籍" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "作者:%(name)s" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "出版社:%(name)s" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "叢書:%(serie)s" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "評分:%(rating)s 星" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "文件格式:%(format)s" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "分類:%(name)s" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "語言:%(name)s" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "下載次數" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "評分列表" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "文件格式列表" -#: cps/web.py:1249 +#: cps/web.py:1259 #, fuzzy msgid "Please configure the SMTP mail settings first..." msgstr "請先配置SMTP郵箱設置..." -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "書籍已經成功加入 %(eReadermail)s 的發送隊列" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "糟糕!發送這本書籍的時候出現錯誤:%(res)s" -#: cps/web.py:1261 +#: cps/web.py:1270 #, fuzzy msgid "Oops! Please update your profile with a valid eReader Email." msgstr "請先設置您的kindle郵箱。" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "註冊" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 #, fuzzy msgid "Connection error to limiter backend, please contact your administrator" msgstr "郵件服務未配置,請聯繫網站管理員!" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "郵件服務未配置,請聯繫網站管理員!" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "您的電子郵件不允許註冊" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "確認郵件已經發送到您的郵箱。" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 #, fuzzy msgid "Cannot activate LDAP authentication" msgstr "無法激活LDAP認證" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, fuzzy, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "您現在已以“%(nickname)s”身份登入" -#: cps/web.py:1407 +#: cps/web.py:1415 #, fuzzy, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "備援登入“%(nickname)s”:無法訪問LDAP伺服器,或用戶未知" -#: cps/web.py:1412 +#: cps/web.py:1420 #, fuzzy, python-format msgid "Could not login: %(message)s" msgstr "無法登入:%(message)s" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 #, fuzzy msgid "Wrong Username or Password" msgstr "用戶名或密碼錯誤" -#: cps/web.py:1423 +#: cps/web.py:1431 #, fuzzy msgid "New Password was sent to your email address" msgstr "新密碼已發送到您的郵箱" -#: cps/web.py:1427 +#: cps/web.py:1435 #, fuzzy msgid "An unknown error occurred. Please try again later." msgstr "發生一個未知錯誤,請稍後再試。" -#: cps/web.py:1429 +#: cps/web.py:1437 #, fuzzy msgid "Please enter valid username to reset password" msgstr "請輸入有效的用戶名進行密碼重置" -#: cps/web.py:1437 +#: cps/web.py:1445 #, fuzzy, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "您現在已以“%(nickname)s”身份登入" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "%(name)s 的用戶配置" -#: cps/web.py:1511 +#: cps/web.py:1526 #, fuzzy msgid "Success! Profile Updated" msgstr "資料已更新" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "使用此郵箱的賬號已經存在。" @@ -1494,54 +1500,58 @@ msgstr "使用此郵箱的賬號已經存在。" msgid "Found no valid gmail.json file with OAuth information" msgstr "找不到包含 OAuth 信息的有效 gmail.json 文件" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, fuzzy, python-format msgid "%(book)s send to E-Reader" msgstr "%(book)s發送到Kindle" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "沒有發現Calibre 電子書轉換器%(tool)s" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "硬碟上找不到 %(format)s 格式" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "發生未知錯誤,書籍轉換失敗" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "Kepubify 轉換失敗:%(error)s" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "找不到轉換後的文件或文件夾%(folder)s中有多個文件" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "電子書轉換器失敗: %(error)s" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "Calibre 運行失敗,錯誤信息:%(error)s" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "電子書轉換器失敗: %(error)s" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1550,30 +1560,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "編輯元數據" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "上傳書籍" @@ -1592,12 +1598,12 @@ msgstr "用戶名" msgid "Email" msgstr "郵箱地址" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 #, fuzzy msgid "Send to eReader Email" msgstr "接收書籍的Kindle郵箱地址" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "管理權限" @@ -1607,8 +1613,8 @@ msgstr "管理權限" msgid "Password" msgstr "密碼" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "下載書籍" @@ -1819,13 +1825,13 @@ msgid "OK" msgstr "確定" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "取消" @@ -1875,16 +1881,76 @@ msgstr "按出版日期排序,最新優先" msgid "Sort according to publishing date, oldest first" msgstr "按出版日期排序,最舊優先" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "減少" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "更多" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "%(range)s 第%(index)s冊" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "語言" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "出版社" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "出版日期" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "簡介:" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "上一個" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "下一個" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "無搜索結果" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "首頁" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "搜索書庫" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "登出" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "刪除書籍" @@ -1913,99 +1979,107 @@ msgstr "轉換到:" msgid "Convert book" msgstr "轉換書籍" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "正在上傳..." + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "關閉" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "錯誤" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "上傳完成,正在處理中,請稍候..." + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "上傳格式" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "書名" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "作者" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "標籤" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "叢書編號" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "出版日期" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "評分" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "簡介" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "書號" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "書號類型" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "書號編號" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "移除" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "添加書號" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "標籤" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "叢書編號" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "評分" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "從URL獲取封面(JPEG - 圖片將下載並存儲在數據庫中)" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "從本地硬碟上傳封面" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "出版日期" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "出版社" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "語言" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "確認" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "沒有" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "上傳格式" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "查看保存書籍" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "獲取元數據" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -2013,38 +2087,32 @@ msgstr "獲取元數據" msgid "Save" msgstr "儲存" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "關鍵字" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 #, fuzzy msgid "Search keyword" msgstr " 搜索關鍵字 " -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "單擊封面將元數據加載到表單" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "加載中..." -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "關閉" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "來源" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "搜索錯誤!" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "無搜索結果!請嘗試另一個關鍵字。" @@ -2153,7 +2221,7 @@ msgid "Enter " msgstr "書號" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "您真的確認?" @@ -2631,74 +2699,61 @@ msgstr "添加顯示或隱藏書籍的標籤值" msgid "Add Allowed/Denied custom column values" msgstr "添加顯示或隱藏書籍的自定義欄位值" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "在線閱讀" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "在線聽書" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "%(range)s 第%(index)s冊" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "出版日期" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "標為未讀" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "標為已讀" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 #, fuzzy msgid "Mark Book as Read or Unread" msgstr "標為未讀" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "已讀" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "從歸檔檔案還原" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "添加到歸檔" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 #, fuzzy msgid "Archive" msgstr "歸檔" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "簡介:" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "添加到書架" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "(公共)" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "編輯元數據" @@ -2771,10 +2826,6 @@ msgstr "輸入網域名" msgid "Denied Domains (Blacklist)" msgstr "禁止註冊的網域名(黑名單)" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "下一個" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "在文本編輯器中打開.kobo/Kobo/Kobo eReader.conf,添加(或編輯):" @@ -2796,11 +2847,16 @@ msgstr "Calibre-Web 實例未配置,請聯繫您的系統管理員!" msgid "Create Issue" msgstr "創建問題" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +#, fuzzy +msgid "Return to Database config" +msgstr "數據庫配置" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "回到首頁" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "登出賬號" @@ -2890,7 +2946,7 @@ msgstr "書籍按評分排序" msgid "Books ordered by file formats" msgstr "書籍按文件格式排序" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "書架列表" @@ -2899,60 +2955,37 @@ msgstr "書架列表" msgid "Books organized in shelves" msgstr "書架上的書" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "首頁" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "切換導航" -#: cps/templates/layout.html:47 -msgid "Search Library" -msgstr "搜索書庫" +#: cps/templates/layout.html:59 +#, fuzzy +msgid "Simple Theme" +msgstr "簡單" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "賬號" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "登出" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "正在上傳..." - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "錯誤" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "上傳完成,正在處理中,請稍候..." - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "設置" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "請不要刷新頁面" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "瀏覽" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "關於" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "上一個" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "書籍詳情" @@ -3068,7 +3101,7 @@ msgstr "父目錄" msgid "Select" msgstr "選擇" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "完成" @@ -3076,35 +3109,81 @@ msgstr "完成" msgid "Calibre-Web eBook Catalog" msgstr "Caliebre-Web電子書路徑" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "epub閱讀器" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +#, fuzzy +msgid "Choose a theme below:" +msgstr "選擇一個用戶名" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "淺色" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "深色" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 #, fuzzy msgid "Black" msgstr "後退" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "打開側欄時重排文本。" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +#, fuzzy +msgid "Default" +msgstr "刪除數據" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +#, fuzzy +msgid "KaiTi" +msgstr "等待中" + +#: cps/templates/read.html:110 +#, fuzzy +msgid "Arial" +msgstr "垂直" + +#: cps/templates/read.html:113 +#, fuzzy +msgid "Spread" +msgstr "已讀" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +#, fuzzy +msgid "One column" +msgstr "無效的閱讀列" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "Comic閱讀器" @@ -3282,10 +3361,6 @@ msgstr "此驗證連接將在10分鐘後失效。" msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "無搜索結果" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "搜索項:" @@ -3302,11 +3377,11 @@ msgstr "出版日期從" msgid "Published Date To" msgstr "出版日期到" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3342,11 +3417,13 @@ msgstr "評分大於" msgid "Rating Below" msgstr "評分小於" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "從:" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "到:" @@ -3370,6 +3447,16 @@ msgstr "禁止改變順序" msgid "Enable Change order" msgstr "允許改變順序" +#: cps/templates/shelf.html:28 +#, fuzzy +msgid "Sort according to book added to shelf, newest first" +msgstr "按圖書日期排序,最新優先" + +#: cps/templates/shelf.html:29 +#, fuzzy +msgid "Sort according to book added to shelf, oldest first" +msgstr "按圖書日期排序,最舊優先" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "書架將被公開" @@ -3438,15 +3525,20 @@ msgstr "任務進度" msgid "Run Time" msgstr "運行時間" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +#, fuzzy +msgid "Message" +msgstr "合併" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3454,6 +3546,10 @@ msgstr "" msgid "Reset user Password" msgstr "重置用戶密碼" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "按語言顯示書籍" diff --git a/cps/ub.py b/cps/ub.py index a9570dd7..20f0ac91 100644 --- a/cps/ub.py +++ b/cps/ub.py @@ -20,7 +20,7 @@ import atexit import os import sys -import datetime +from datetime import datetime, timezone, timedelta import itertools import uuid from flask import session as flask_session @@ -54,7 +54,7 @@ from sqlalchemy.orm import backref, relationship, sessionmaker, Session, scoped_ from werkzeug.security import generate_password_hash from . import constants, logger - +from .string_helper import strip_whitespaces log = logger.create() @@ -77,7 +77,7 @@ def store_user_session(): if flask_session.get('_user_id', ""): try: if not check_user_session(_user, _id, _random): - expiry = int((datetime.datetime.now() + datetime.timedelta(days=31)).timestamp()) + expiry = int((datetime.now() + timedelta(days=31)).timestamp()) user_session = User_Sessions(_user, _id, _random, expiry) session.add(user_session) session.commit() @@ -109,7 +109,7 @@ def check_user_session(user_id, session_key, random): User_Sessions.random == random, ).one_or_none() if found is not None: - new_expiry = int((datetime.datetime.now() + datetime.timedelta(days=31)).timestamp()) + new_expiry = int((datetime.now() + timedelta(days=31)).timestamp()) if new_expiry - found.expiry > 86400: found.expiry = new_expiry session.merge(found) @@ -196,19 +196,19 @@ class UserBase: def list_denied_tags(self): mct = self.denied_tags or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def list_allowed_tags(self): mct = self.allowed_tags or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def list_denied_column_values(self): mct = self.denied_column_value or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def list_allowed_column_values(self): mct = self.allowed_column_value or "" - return [t.strip() for t in mct.split(",")] + return [strip_whitespaces(t) for t in mct.split(",")] def get_view_property(self, page, prop): if not self.view_settings.get(page): @@ -233,7 +233,7 @@ class UserBase: return '' % self.name -# Baseclass for Users in Calibre-Web, settings which are depending on certain users are stored here. It is derived from +# Baseclass for Users in Calibre-Web, settings which depend on certain users are stored here. It is derived from # User Base (all access methods are declared there) class User(UserBase, Base): __tablename__ = 'user' @@ -311,7 +311,6 @@ class Anonymous(AnonymousUserMixin, UserBase): self.view_settings = data.view_settings self.kobo_only_shelves_sync = data.kobo_only_shelves_sync - def role_admin(self): return False @@ -370,8 +369,8 @@ class Shelf(Base): user_id = Column(Integer, ForeignKey('user.id')) kobo_sync = Column(Boolean, default=False) books = relationship("BookShelf", backref="ub_shelf", cascade="all, delete-orphan", lazy="dynamic") - created = Column(DateTime, default=datetime.datetime.utcnow) - last_modified = Column(DateTime, default=datetime.datetime.utcnow, onupdate=datetime.datetime.utcnow) + created = Column(DateTime, default=lambda: datetime.now(timezone.utc)) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc), onupdate=lambda: datetime.now(timezone.utc)) def __repr__(self): return '' % (self.id, self.name) @@ -385,7 +384,7 @@ class BookShelf(Base): book_id = Column(Integer) order = Column(Integer) shelf = Column(Integer, ForeignKey('shelf.id')) - date_added = Column(DateTime, default=datetime.datetime.utcnow) + date_added = Column(DateTime, default=lambda: datetime.now(timezone.utc)) def __repr__(self): return '' % self.id @@ -398,7 +397,7 @@ class ShelfArchive(Base): id = Column(Integer, primary_key=True) uuid = Column(String) user_id = Column(Integer, ForeignKey('user.id')) - last_modified = Column(DateTime, default=datetime.datetime.utcnow) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc)) class ReadBook(Base): @@ -418,7 +417,7 @@ class ReadBook(Base): cascade="all", backref=backref("book_read_link", uselist=False)) - last_modified = Column(DateTime, default=datetime.datetime.utcnow, onupdate=datetime.datetime.utcnow) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc), onupdate=lambda: datetime.now(timezone.utc)) last_time_started_reading = Column(DateTime, nullable=True) times_started_reading = Column(Integer, default=0, nullable=False) @@ -441,7 +440,7 @@ class ArchivedBook(Base): user_id = Column(Integer, ForeignKey('user.id')) book_id = Column(Integer) is_archived = Column(Boolean, unique=False) - last_modified = Column(DateTime, default=datetime.datetime.utcnow) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc)) class KoboSyncedBooks(Base): @@ -460,8 +459,8 @@ class KoboReadingState(Base): id = Column(Integer, primary_key=True, autoincrement=True) user_id = Column(Integer, ForeignKey('user.id')) book_id = Column(Integer) - last_modified = Column(DateTime, default=datetime.datetime.utcnow, onupdate=datetime.datetime.utcnow) - priority_timestamp = Column(DateTime, default=datetime.datetime.utcnow, onupdate=datetime.datetime.utcnow) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc), onupdate=lambda: datetime.now(timezone.utc)) + priority_timestamp = Column(DateTime, default=lambda: datetime.now(timezone.utc), onupdate=lambda: datetime.now(timezone.utc)) current_bookmark = relationship("KoboBookmark", uselist=False, backref="kobo_reading_state", cascade="all, delete") statistics = relationship("KoboStatistics", uselist=False, backref="kobo_reading_state", cascade="all, delete") @@ -471,7 +470,7 @@ class KoboBookmark(Base): id = Column(Integer, primary_key=True) kobo_reading_state_id = Column(Integer, ForeignKey('kobo_reading_state.id')) - last_modified = Column(DateTime, default=datetime.datetime.utcnow, onupdate=datetime.datetime.utcnow) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc), onupdate=lambda: datetime.now(timezone.utc)) location_source = Column(String) location_type = Column(String) location_value = Column(String) @@ -484,7 +483,7 @@ class KoboStatistics(Base): id = Column(Integer, primary_key=True) kobo_reading_state_id = Column(Integer, ForeignKey('kobo_reading_state.id')) - last_modified = Column(DateTime, default=datetime.datetime.utcnow, onupdate=datetime.datetime.utcnow) + last_modified = Column(DateTime, default=lambda: datetime.now(timezone.utc), onupdate=lambda: datetime.now(timezone.utc)) remaining_time_minutes = Column(Integer) spent_reading_minutes = Column(Integer) @@ -495,11 +494,11 @@ def receive_before_flush(session, flush_context, instances): for change in itertools.chain(session.new, session.dirty): if isinstance(change, (ReadBook, KoboStatistics, KoboBookmark)): if change.kobo_reading_state: - change.kobo_reading_state.last_modified = datetime.datetime.utcnow() - # Maintain the last_modified bit for the Shelf table. + change.kobo_reading_state.last_modified = datetime.now(timezone.utc) + # Maintain the last_modified_bit for the Shelf table. for change in itertools.chain(session.new, session.deleted): if isinstance(change, BookShelf): - change.ub_shelf.last_modified = datetime.datetime.utcnow() + change.ub_shelf.last_modified = datetime.now(timezone.utc) # Baseclass representing Downloads from calibre-web in app.db @@ -539,7 +538,7 @@ class RemoteAuthToken(Base): def __init__(self): super().__init__() self.auth_token = (hexlify(os.urandom(4))).decode('utf-8') - self.expiration = datetime.datetime.now() + datetime.timedelta(minutes=10) # 10 min from now + self.expiration = datetime.now() + timedelta(minutes=10) # 10 min from now def __repr__(self): return '' % self.id @@ -563,7 +562,7 @@ class Thumbnail(Base): type = Column(SmallInteger, default=constants.THUMBNAIL_TYPE_COVER) resolution = Column(SmallInteger, default=constants.COVER_THUMBNAIL_SMALL) filename = Column(String, default=filename) - generated_at = Column(DateTime, default=lambda: datetime.datetime.utcnow()) + generated_at = Column(DateTime, default=lambda: datetime.now(timezone.utc)) expiration = Column(DateTime, nullable=True) @@ -602,7 +601,7 @@ def migrate_user_session_table(engine, _session): trans.commit() -# Migrate database to current version, has to be updated after every database change. Currently migration from +# Migrate database to current version, has to be updated after every database change. Currently, migration from # maybe 4/5 versions back to current should work. # Migration is done by checking if relevant columns are existing, and then adding rows with SQL commands def migrate_Database(_session): @@ -614,7 +613,7 @@ def migrate_Database(_session): def clean_database(_session): # Remove expired remote login tokens - now = datetime.datetime.now() + now = datetime.now() try: _session.query(RemoteAuthToken).filter(now > RemoteAuthToken.expiration).\ filter(RemoteAuthToken.token_type != 1).delete() @@ -637,7 +636,7 @@ def update_download(book_id, user_id): session.rollback() -# Delete non existing downloaded books in calibre-web's own database +# Delete non-existing downloaded books in calibre-web's own database def delete_download(book_id): session.query(Downloads).filter(book_id == Downloads.book_id).delete() try: diff --git a/cps/updater.py b/cps/updater.py index ab67e7a6..e33b80f4 100644 --- a/cps/updater.py +++ b/cps/updater.py @@ -329,7 +329,7 @@ class Updater(threading.Thread): @classmethod def _stable_version_info(cls): log.debug("Stable version: {}".format(constants.STABLE_VERSION)) - return constants.STABLE_VERSION # Current Version + return {'version': constants.STABLE_VERSION } @classmethod def dry_run(cls): diff --git a/cps/uploader.py b/cps/uploader.py index bb3a49b7..b991049b 100644 --- a/cps/uploader.py +++ b/cps/uploader.py @@ -23,7 +23,8 @@ from flask_babel import gettext as _ from . import logger, comic, isoLanguages from .constants import BookMeta from .helper import split_authors -from .file_helper import get_temp_dir, validate_mime_type +from .file_helper import get_temp_dir +from .string_helper import strip_whitespaces log = logger.create() @@ -38,17 +39,20 @@ except (ImportError, RuntimeError) as e: try: from pypdf import PdfReader + from pypdf.generic import NullObject use_pdf_meta = True except ImportError as ex: log.debug('PyPDF is recommended for best performance in metadata extracting from pdf files: %s', ex) try: from PyPDF2 import PdfReader + from pypdf.generic import NullObject use_pdf_meta = True except ImportError as ex: log.debug('PyPDF is recommended for best performance in metadata extracting from pdf files: %s', ex) log.debug('PyPdf2 is also possible for metadata extracting from pdf files, but not recommended anymore') try: from PyPDF3 import PdfFileReader as PdfReader + from pypdf.generic import NullObject use_pdf_meta = True except ImportError as e: log.debug('Cannot import PyPDF3/PyPDF2, extracting pdf metadata will not work: %s / %s', e) @@ -68,28 +72,39 @@ except ImportError as e: log.debug('Cannot import fb2, extracting fb2 metadata will not work: %s', e) use_fb2_meta = False +try: + from . import audio + use_audio_meta = True +except ImportError as e: + log.debug('Cannot import mutagen, extracting audio metadata will not work: %s', e) + use_audio_meta = False -def process(tmp_file_path, original_file_name, original_file_extension, rar_executable): + +def process(tmp_file_path, original_file_name, original_file_extension, rar_executable, no_cover=False): meta = default_meta(tmp_file_path, original_file_name, original_file_extension) extension_upper = original_file_extension.upper() try: if ".PDF" == extension_upper: - meta = pdf_meta(tmp_file_path, original_file_name, original_file_extension) + meta = pdf_meta(tmp_file_path, original_file_name, original_file_extension, no_cover) elif extension_upper in [".KEPUB", ".EPUB"] and use_epub_meta is True: - meta = epub.get_epub_info(tmp_file_path, original_file_name, original_file_extension) + meta = epub.get_epub_info(tmp_file_path, original_file_name, original_file_extension, no_cover) elif ".FB2" == extension_upper and use_fb2_meta is True: meta = fb2.get_fb2_info(tmp_file_path, original_file_extension) elif extension_upper in ['.CBZ', '.CBT', '.CBR', ".CB7"]: meta = comic.get_comic_info(tmp_file_path, original_file_name, original_file_extension, - rar_executable) + rar_executable, + no_cover) + elif extension_upper in [".MP3", ".OGG", ".FLAC", ".WAV", ".AAC", ".AIFF", ".ASF", ".MP4", + ".M4A", ".M4B", ".OGV", ".OPUS"] and use_audio_meta: + meta = audio.get_audio_file_info(tmp_file_path, original_file_extension, original_file_name, no_cover) except Exception as ex: log.warning('cannot parse metadata, using default: %s', ex) - if not meta.title.strip(): + if not strip_whitespaces(meta.title): meta = meta._replace(title=original_file_name) - if not meta.author.strip() or meta.author.lower() == 'unknown': + if not strip_whitespaces(meta.author) or meta.author.lower() == 'unknown': meta = meta._replace(author=_('Unknown')) return meta @@ -157,7 +172,7 @@ def parse_xmp(pdf_file): } -def pdf_meta(tmp_file_path, original_file_name, original_file_extension): +def pdf_meta(tmp_file_path, original_file_name, original_file_extension, no_cover_processing): doc_info = None xmp_info = None @@ -193,10 +208,12 @@ def pdf_meta(tmp_file_path, original_file_name, original_file_extension): if subject == '': subject = doc_info.subject or "" if tags == '' and '/Keywords' in doc_info: - if isinstance(doc_info['/Keywords'], bytes): - tags = doc_info['/Keywords'].decode('utf-8') - else: - tags = doc_info['/Keywords'] + keywords = doc_info['/Keywords'] + if not isinstance(keywords, NullObject): + if isinstance(keywords, bytes): + tags = keywords.decode('utf-8') + else: + tags = keywords else: title = original_file_name @@ -205,7 +222,7 @@ def pdf_meta(tmp_file_path, original_file_name, original_file_extension): extension=original_file_extension, title=title, author=author, - cover=pdf_preview(tmp_file_path, original_file_name), + cover=pdf_preview(tmp_file_path, original_file_name) if not no_cover_processing else None, description=subject, tags=tags, series="", @@ -220,7 +237,7 @@ def pdf_preview(tmp_file_path, tmp_dir): if use_generic_pdf_cover: return None try: - cover_file_name = os.path.splitext(tmp_file_path)[0] + ".cover.jpg" + cover_file_name = tmp_file_path + ".jpg" with Image() as img: img.options["pdf:use-cropbox"] = "true" img.read(filename=tmp_file_path + '[0]', resolution=150) @@ -228,7 +245,7 @@ def pdf_preview(tmp_file_path, tmp_dir): if img.alpha_channel: img.alpha_channel = 'remove' img.background_color = Color('white') - img.save(filename=os.path.join(tmp_dir, cover_file_name)) + img.save(filename=cover_file_name) return cover_file_name except PolicyError as ex: log.warning('Pdf extraction forbidden by Imagemagick policy: %s', ex) diff --git a/cps/web.py b/cps/web.py index f43cdd87..31e124e9 100644 --- a/cps/web.py +++ b/cps/web.py @@ -23,9 +23,9 @@ import json import mimetypes import chardet # dependency of requests import copy +from importlib.metadata import metadata -from flask import Blueprint, jsonify -from flask import request, redirect, send_from_directory, make_response, flash, abort, url_for, Response +from flask import Blueprint, jsonify, request, redirect, send_from_directory, make_response, flash, abort, url_for from flask import session as flask_session from flask_babel import gettext as _ from flask_babel import get_locale @@ -36,7 +36,6 @@ from sqlalchemy.exc import IntegrityError, InvalidRequestError, OperationalError from sqlalchemy.sql.expression import text, func, false, not_, and_, or_ from sqlalchemy.orm.attributes import flag_modified from sqlalchemy.sql.functions import coalesce - from werkzeug.datastructures import Headers from werkzeug.security import generate_password_hash, check_password_hash @@ -51,7 +50,7 @@ from .helper import check_valid_domain, check_email, check_username, \ edit_book_read_status, valid_password from .pagination import Pagination from .redirect import get_redirect_location -from .babel import get_available_locale +from .cw_babel import get_available_locale from .usermanagement import login_required_if_no_ano from .kobo_sync_status import remove_synced_book from .render_template import render_title_template @@ -60,6 +59,7 @@ from . import limiter from .services.worker import WorkerThread from .tasks_status import render_task_status from .usermanagement import user_login_required +from .string_helper import strip_whitespaces feature_support = { @@ -85,6 +85,10 @@ except ImportError: sort = sorted # Just use regular sort then, may cause issues with badly named pages in cbz/cbr files +sql_version = metadata("sqlalchemy")["Version"] +sqlalchemy_version2 = ([int(x) if x.isnumeric() else 0 for x in sql_version.split('.')[:3]] >= [2, 0, 0]) + + @app.after_request def add_security_headers(resp): default_src = ([host.strip() for host in config.config_trustedhosts.split(',') if host] + @@ -108,7 +112,7 @@ def add_security_headers(resp): resp.headers['X-Content-Type-Options'] = 'nosniff' resp.headers['X-Frame-Options'] = 'SAMEORIGIN' resp.headers['X-XSS-Protection'] = '1; mode=block' - resp.headers['Strict-Transport-Security'] = 'max-age=31536000'; + resp.headers['Strict-Transport-Security'] = 'max-age=31536000' return resp @@ -298,9 +302,10 @@ def get_languages_json(): def get_matching_tags(): tag_dict = {'tags': []} q = calibre_db.session.query(db.Books).filter(calibre_db.common_filters(True)) - calibre_db.session.connection().connection.connection.create_function("lower", 1, db.lcase) - author_input = request.args.get('author_name') or '' - title_input = request.args.get('book_title') or '' + calibre_db.create_functions() + # calibre_db.session.connection().connection.connection.create_function("lower", 1, db.lcase) + author_input = request.args.get('authors') or '' + title_input = request.args.get('title') or '' include_tag_inputs = request.args.getlist('include_tag') or '' exclude_tag_inputs = request.args.getlist('exclude_tag') or '' q = q.filter(db.Books.authors.any(func.lower(db.Authors.name).ilike("%" + author_input + "%")), @@ -530,7 +535,7 @@ def render_author_books(page, author_id, order): flash(_("Oops! Selected book is unavailable. File does not exist or is not accessible"), category="error") return redirect(url_for("web.index")) - if constants.sqlalchemy_version2: + if sqlalchemy_version2: author = calibre_db.session.get(db.Authors, author_id) else: author = calibre_db.session.query(db.Authors).get(author_id) @@ -790,7 +795,7 @@ def render_archived_books(page, sort_param): True, True, config.config_read_column) - name = _('Archived Books') + ' (' + str(len(archived_book_ids)) + ')' + name = _('Archived Books') + ' (' + str(len(entries)) + ')' page_name = "archived" return render_title_template('index.html', random=random, entries=entries, pagination=pagination, title=name, page=page_name, order=sort_param[1]) @@ -1192,13 +1197,14 @@ def serve_book(book_id, book_format, anyname): if not data: return "File not in Database" range_header = request.headers.get('Range', None) - + if not range_header: + log.info('Serving book: \'%s\' to %s - %s', data.name, current_user.name, + request.headers.get('X-Forwarded-For', request.remote_addr)) if config.config_use_google_drive: try: headers = Headers() headers["Content-Type"] = mimetypes.types_map.get('.' + book_format, "application/octet-stream") - if not range_header: - log.info('Serving book: %s', data.name) + if not range_header: headers['Accept-Ranges'] = 'bytes' df = getFileFromEbooksFolder(book.path, data.name + "." + book_format) return do_gdrive_download(df, headers, (book_format.upper() == 'TXT')) @@ -1207,7 +1213,6 @@ def serve_book(book_id, book_format, anyname): return "File Not Found" else: if book_format.upper() == 'TXT': - log.info('Serving book: %s', data.name) try: rawdata = open(os.path.join(config.get_book_path(), book.path, data.name + "." + book_format), "rb").read() @@ -1228,7 +1233,6 @@ def serve_book(book_id, book_format, anyname): response = make_response( send_from_directory(os.path.join(config.get_book_path(), book.path), data.name + "." + book_format)) if not range_header: - log.info('Serving book: %s', data.name) response.headers['Accept-Ranges'] = 'bytes' return response @@ -1238,7 +1242,12 @@ def serve_book(book_id, book_format, anyname): @login_required_if_no_ano @download_required def download_link(book_id, book_format, anyname): - client = "kobo" if "Kobo" in request.headers.get('User-Agent') else "" + if "kindle" in request.headers.get('User-Agent').lower(): + client = "kindle" + elif "Kobo" in request.headers.get('User-Agent').lower(): + client = "kobo" + else: + client = "" return get_download_link(book_id, book_format, client) @@ -1247,8 +1256,7 @@ def download_link(book_id, book_format, anyname): @download_required def send_to_ereader(book_id, book_format, convert): if not config.get_mail_server_configured(): - response = [{'type': "danger", 'message': _("Please configure the SMTP mail settings first...")}] - return Response(json.dumps(response), mimetype='application/json') + return make_response(jsonify(type="danger", message=_("Please configure the SMTP mail settings first..."))) elif current_user.kindle_mail: result = send_mail(book_id, book_format, convert, current_user.kindle_mail, config.get_book_path(), current_user.name) @@ -1260,7 +1268,7 @@ def send_to_ereader(book_id, book_format, convert): response = [{'type': "danger", 'message': _("Oops! There was an error sending book: %(res)s", res=result)}] else: response = [{'type': "danger", 'message': _("Oops! Please update your profile with a valid eReader Email.")}] - return Response(json.dumps(response), mimetype='application/json') + return make_response(jsonify(response)) # ################################### Login Logout ################################################################## @@ -1286,7 +1294,7 @@ def register_post(): if not config.get_mail_server_configured(): flash(_("Oops! Email server is not configured, please contact your administrator."), category="error") return render_title_template('register.html', title=_("Register"), page="register") - nickname = to_save.get("email", "").strip() if config.config_register_email else to_save.get('name') + nickname = strip_whitespaces(to_save.get("email", "")) if config.config_register_email else to_save.get('name') if not nickname or not to_save.get("email"): flash(_("Oops! Please complete all fields."), category="error") return render_title_template('register.html', title=_("Register"), page="register") @@ -1311,7 +1319,7 @@ def register_post(): ub.session.commit() if feature_support['oauth']: register_user_with_oauth(content) - send_registration_mail(to_save.get("email", "").strip(), nickname, password) + send_registration_mail(strip_whitespaces(to_save.get("email", "")), nickname, password) except Exception: ub.session.rollback() flash(_("Oops! An unknown error occurred. Please try again later."), category="error") @@ -1370,11 +1378,11 @@ def login(): @web.route('/login', methods=['POST']) -@limiter.limit("40/day", key_func=lambda: request.form.get('username', "").strip().lower()) -@limiter.limit("3/minute", key_func=lambda: request.form.get('username', "").strip().lower()) +@limiter.limit("40/day", key_func=lambda: strip_whitespaces(request.form.get('username', "")).lower()) +@limiter.limit("3/minute", key_func=lambda: strip_whitespaces(request.form.get('username', "")).lower()) def login_post(): form = request.form.to_dict() - username = form.get('username', "").strip().lower().replace("\n","").replace("\r","") + username = strip_whitespaces(form.get('username', "")).lower().replace("\n","").replace("\r","") try: limiter.check() except RateLimitExceeded: @@ -1578,9 +1586,10 @@ def read_book(book_id, book_format): bookmark = ub.session.query(ub.Bookmark).filter(and_(ub.Bookmark.user_id == int(current_user.id), ub.Bookmark.book_id == book_id, ub.Bookmark.format == book_format.upper())).first() - if book_format.lower() == "epub": - log.debug("Start epub reader for %d", book_id) - return render_title_template('read.html', bookid=book_id, title=book.title, bookmark=bookmark) + if book_format.lower() == "epub" or book_format.lower() == "kepub": + log.debug("Start [k]epub reader for %d", book_id) + return render_title_template('read.html', bookid=book_id, title=book.title, bookmark=bookmark, + book_format=book_format) elif book_format.lower() == "pdf": log.debug("Start pdf reader for %d", book_id) return render_title_template('readpdf.html', pdffile=book_id, title=book.title) @@ -1641,6 +1650,11 @@ def show_book(book_id): entry.email_share_list = check_send_to_ereader(entry) entry.reader_list = check_read_formats(entry) + entry.reader_list_sizes = dict() + for data in entry.data: + if data.format.lower() in entry.reader_list: + entry.reader_list_sizes[data.format.lower()] = data.uncompressed_size + entry.audio_entries = [] for media_format in entry.data: if media_format.format.lower() in constants.EXTENSIONS_AUDIO: diff --git a/messages.pot b/messages.pot index c47abff7..646f3cc7 100644 --- a/messages.pot +++ b/messages.pot @@ -1,511 +1,529 @@ # Translations template for PROJECT. -# Copyright (C) 2024 ORGANIZATION +# Copyright (C) 2025 ORGANIZATION # This file is distributed under the same license as the PROJECT project. -# FIRST AUTHOR , 2024. +# FIRST AUTHOR , 2025. # #, fuzzy msgid "" msgstr "" "Project-Id-Version: PROJECT VERSION\n" "Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" -"POT-Creation-Date: 2024-07-07 10:17+0200\n" +"POT-Creation-Date: 2025-03-30 15:55+0200\n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" "Last-Translator: FULL NAME \n" "Language-Team: LANGUAGE \n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" -"Generated-By: Babel 2.13.1\n" +"Generated-By: Babel 2.15.0\n" -#: cps/about.py:84 +#: cps/about.py:85 msgid "Statistics" msgstr "" -#: cps/admin.py:150 +#: cps/admin.py:151 msgid "Server restarted, please reload page." msgstr "" -#: cps/admin.py:152 +#: cps/admin.py:153 msgid "Performing Server shutdown, please close window." msgstr "" -#: cps/admin.py:160 +#: cps/admin.py:161 msgid "Success! Database Reconnected" msgstr "" -#: cps/admin.py:163 +#: cps/admin.py:164 msgid "Unknown command" msgstr "" -#: cps/admin.py:174 +#: cps/admin.py:175 msgid "Success! Books queued for Metadata Backup, please check Tasks for result" msgstr "" -#: cps/admin.py:207 cps/editbooks.py:573 cps/editbooks.py:575 -#: cps/editbooks.py:615 cps/editbooks.py:658 cps/editbooks.py:1280 -#: cps/updater.py:615 cps/uploader.py:93 cps/uploader.py:102 +#: cps/admin.py:208 cps/editbooks.py:614 cps/editbooks.py:657 +#: cps/editbooks.py:1302 cps/updater.py:615 cps/uploader.py:108 +#: cps/uploader.py:117 msgid "Unknown" msgstr "" -#: cps/admin.py:232 +#: cps/admin.py:233 msgid "Admin page" msgstr "" -#: cps/admin.py:252 +#: cps/admin.py:253 msgid "Basic Configuration" msgstr "" -#: cps/admin.py:290 +#: cps/admin.py:291 msgid "UI Configuration" msgstr "" -#: cps/admin.py:324 cps/templates/admin.html:51 +#: cps/admin.py:315 cps/admin.py:996 cps/db.py:793 cps/search.py:150 +#: cps/web.py:753 +#, python-format +msgid "Custom Column No.%(column)d does not exist in calibre database" +msgstr "" + +#: cps/admin.py:333 cps/templates/admin.html:51 msgid "Edit Users" msgstr "" -#: cps/admin.py:368 cps/opds.py:542 cps/templates/grid.html:14 +#: cps/admin.py:377 cps/opds.py:540 cps/templates/grid.html:14 #: cps/templates/list.html:13 msgid "All" msgstr "" -#: cps/admin.py:395 cps/admin.py:1414 +#: cps/admin.py:401 cps/admin.py:1426 msgid "User not found" msgstr "" -#: cps/admin.py:409 +#: cps/admin.py:415 msgid "{} users deleted successfully" msgstr "" -#: cps/admin.py:432 cps/templates/config_view_edit.html:133 +#: cps/admin.py:438 cps/templates/config_view_edit.html:133 #: cps/templates/user_edit.html:45 cps/templates/user_table.html:81 msgid "Show All" msgstr "" -#: cps/admin.py:453 cps/admin.py:459 +#: cps/admin.py:459 cps/admin.py:465 msgid "Malformed request" msgstr "" -#: cps/admin.py:471 cps/admin.py:2046 +#: cps/admin.py:477 cps/admin.py:2069 msgid "Guest Name can't be changed" msgstr "" -#: cps/admin.py:483 +#: cps/admin.py:489 msgid "Guest can't have this role" msgstr "" -#: cps/admin.py:495 cps/admin.py:2000 +#: cps/admin.py:501 cps/admin.py:2023 msgid "No admin user remaining, can't remove admin role" msgstr "" -#: cps/admin.py:499 cps/admin.py:513 +#: cps/admin.py:505 cps/admin.py:519 msgid "Value has to be true or false" msgstr "" -#: cps/admin.py:501 +#: cps/admin.py:507 msgid "Invalid role" msgstr "" -#: cps/admin.py:505 +#: cps/admin.py:511 msgid "Guest can't have this view" msgstr "" -#: cps/admin.py:515 +#: cps/admin.py:521 msgid "Invalid view" msgstr "" -#: cps/admin.py:518 +#: cps/admin.py:524 msgid "Guest's Locale is determined automatically and can't be set" msgstr "" -#: cps/admin.py:522 +#: cps/admin.py:528 msgid "No Valid Locale Given" msgstr "" -#: cps/admin.py:533 +#: cps/admin.py:539 msgid "No Valid Book Language Given" msgstr "" -#: cps/admin.py:535 cps/editbooks.py:440 +#: cps/admin.py:541 cps/editbooks.py:292 msgid "Parameter not found" msgstr "" -#: cps/admin.py:572 +#: cps/admin.py:578 msgid "Invalid Read Column" msgstr "" -#: cps/admin.py:578 +#: cps/admin.py:584 msgid "Invalid Restricted Column" msgstr "" -#: cps/admin.py:598 cps/admin.py:1871 +#: cps/admin.py:604 cps/admin.py:1894 msgid "Calibre-Web configuration updated" msgstr "" -#: cps/admin.py:610 +#: cps/admin.py:616 msgid "Do you really want to delete the Kobo Token?" msgstr "" -#: cps/admin.py:612 +#: cps/admin.py:618 msgid "Do you really want to delete this domain?" msgstr "" -#: cps/admin.py:614 +#: cps/admin.py:620 msgid "Do you really want to delete this user?" msgstr "" -#: cps/admin.py:616 +#: cps/admin.py:622 msgid "Are you sure you want to delete this shelf?" msgstr "" -#: cps/admin.py:618 +#: cps/admin.py:624 msgid "Are you sure you want to change locales of selected user(s)?" msgstr "" -#: cps/admin.py:620 +#: cps/admin.py:626 msgid "Are you sure you want to change visible book languages for selected user(s)?" msgstr "" -#: cps/admin.py:622 +#: cps/admin.py:628 msgid "Are you sure you want to change the selected role for the selected user(s)?" msgstr "" -#: cps/admin.py:624 +#: cps/admin.py:630 msgid "Are you sure you want to change the selected restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:626 +#: cps/admin.py:632 msgid "Are you sure you want to change the selected visibility restrictions for the selected user(s)?" msgstr "" -#: cps/admin.py:629 +#: cps/admin.py:635 msgid "Are you sure you want to change shelf sync behavior for the selected user(s)?" msgstr "" -#: cps/admin.py:631 +#: cps/admin.py:637 msgid "Are you sure you want to change Calibre library location?" msgstr "" -#: cps/admin.py:633 +#: cps/admin.py:639 msgid "Calibre-Web will search for updated Covers and update Cover Thumbnails, this may take a while?" msgstr "" -#: cps/admin.py:636 +#: cps/admin.py:642 msgid "Are you sure you want delete Calibre-Web's sync database to force a full sync with your Kobo Reader?" msgstr "" -#: cps/admin.py:879 cps/admin.py:885 cps/admin.py:895 cps/admin.py:905 +#: cps/admin.py:885 cps/admin.py:891 cps/admin.py:901 cps/admin.py:911 #: cps/templates/modal_dialogs.html:29 cps/templates/user_table.html:41 #: cps/templates/user_table.html:58 msgid "Deny" msgstr "" -#: cps/admin.py:881 cps/admin.py:887 cps/admin.py:897 cps/admin.py:907 +#: cps/admin.py:887 cps/admin.py:893 cps/admin.py:903 cps/admin.py:913 #: cps/templates/modal_dialogs.html:28 cps/templates/user_table.html:44 #: cps/templates/user_table.html:61 msgid "Allow" msgstr "" -#: cps/admin.py:940 +#: cps/admin.py:946 msgid "{} sync entries deleted" msgstr "" -#: cps/admin.py:981 +#: cps/admin.py:987 msgid "Tag not found" msgstr "" -#: cps/admin.py:993 +#: cps/admin.py:1005 msgid "Invalid Action" msgstr "" -#: cps/admin.py:1120 +#: cps/admin.py:1132 msgid "client_secrets.json Is Not Configured For Web Application" msgstr "" -#: cps/admin.py:1165 +#: cps/admin.py:1177 msgid "Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1171 +#: cps/admin.py:1183 msgid "Access Logfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1205 +#: cps/admin.py:1217 msgid "Please Enter a LDAP Provider, Port, DN and User Object Identifier" msgstr "" -#: cps/admin.py:1211 +#: cps/admin.py:1223 msgid "Please Enter a LDAP Service Account and Password" msgstr "" -#: cps/admin.py:1214 +#: cps/admin.py:1226 msgid "Please Enter a LDAP Service Account" msgstr "" -#: cps/admin.py:1219 +#: cps/admin.py:1231 #, python-format msgid "LDAP Group Object Filter Needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1221 +#: cps/admin.py:1233 msgid "LDAP Group Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1225 +#: cps/admin.py:1237 #, python-format msgid "LDAP User Object Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1227 +#: cps/admin.py:1239 msgid "LDAP User Object Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1234 +#: cps/admin.py:1246 #, python-format msgid "LDAP Member User Filter needs to Have One \"%s\" Format Identifier" msgstr "" -#: cps/admin.py:1236 +#: cps/admin.py:1248 msgid "LDAP Member User Filter Has Unmatched Parenthesis" msgstr "" -#: cps/admin.py:1243 +#: cps/admin.py:1255 msgid "LDAP CACertificate, Certificate or Key Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1274 cps/templates/admin.html:53 +#: cps/admin.py:1286 cps/templates/admin.html:53 msgid "Add New User" msgstr "" -#: cps/admin.py:1283 cps/templates/admin.html:100 +#: cps/admin.py:1295 cps/templates/admin.html:100 msgid "Edit Email Server Settings" msgstr "" -#: cps/admin.py:1302 +#: cps/admin.py:1314 msgid "Success! Gmail Account Verified." msgstr "" -#: cps/admin.py:1322 cps/admin.py:1325 cps/admin.py:1710 cps/admin.py:1855 -#: cps/admin.py:1953 cps/admin.py:2074 cps/editbooks.py:226 -#: cps/editbooks.py:303 cps/editbooks.py:1242 cps/shelf.py:90 cps/shelf.py:150 +#: cps/admin.py:1334 cps/admin.py:1337 cps/admin.py:1722 cps/admin.py:1878 +#: cps/admin.py:1976 cps/admin.py:2097 cps/editbooks.py:168 +#: cps/editbooks.py:561 cps/editbooks.py:1256 cps/shelf.py:90 cps/shelf.py:150 #: cps/shelf.py:193 cps/shelf.py:243 cps/shelf.py:280 cps/shelf.py:354 -#: cps/shelf.py:468 cps/tasks/convert.py:153 cps/web.py:1520 +#: cps/shelf.py:476 cps/tasks/convert.py:160 cps/web.py:1535 #, python-format msgid "Oops! Database Error: %(error)s." msgstr "" -#: cps/admin.py:1332 +#: cps/admin.py:1344 #, python-format msgid "Test e-mail queued for sending to %(email)s, please check Tasks for result" msgstr "" -#: cps/admin.py:1335 +#: cps/admin.py:1347 #, python-format msgid "There was an error sending the Test e-mail: %(res)s" msgstr "" -#: cps/admin.py:1337 +#: cps/admin.py:1349 msgid "Please configure your e-mail address first..." msgstr "" -#: cps/admin.py:1339 +#: cps/admin.py:1351 msgid "Email Server Settings updated" msgstr "" -#: cps/admin.py:1362 cps/templates/admin.html:195 +#: cps/admin.py:1374 cps/templates/admin.html:195 msgid "Edit Scheduled Tasks Settings" msgstr "" -#: cps/admin.py:1374 +#: cps/admin.py:1386 msgid "Invalid start time for task specified" msgstr "" -#: cps/admin.py:1379 +#: cps/admin.py:1391 msgid "Invalid duration for task specified" msgstr "" -#: cps/admin.py:1389 +#: cps/admin.py:1401 msgid "Scheduled tasks settings updated" msgstr "" -#: cps/admin.py:1399 cps/admin.py:1448 cps/admin.py:2070 cps/web.py:1316 +#: cps/admin.py:1411 cps/admin.py:1460 cps/admin.py:2093 cps/web.py:1325 msgid "Oops! An unknown error occurred. Please try again later." msgstr "" -#: cps/admin.py:1403 +#: cps/admin.py:1415 msgid "Settings DB is not Writeable" msgstr "" -#: cps/admin.py:1433 cps/admin.py:2062 +#: cps/admin.py:1445 cps/admin.py:2085 #, python-format msgid "Edit User %(nick)s" msgstr "" -#: cps/admin.py:1445 +#: cps/admin.py:1457 #, python-format msgid "Success! Password for user %(user)s reset" msgstr "" -#: cps/admin.py:1451 +#: cps/admin.py:1463 msgid "Oops! Please configure the SMTP mail settings." msgstr "" -#: cps/admin.py:1462 +#: cps/admin.py:1474 msgid "Logfile viewer" msgstr "" -#: cps/admin.py:1528 +#: cps/admin.py:1540 msgid "Requesting update package" msgstr "" -#: cps/admin.py:1529 +#: cps/admin.py:1541 msgid "Downloading update package" msgstr "" -#: cps/admin.py:1530 +#: cps/admin.py:1542 msgid "Unzipping update package" msgstr "" -#: cps/admin.py:1531 +#: cps/admin.py:1543 msgid "Replacing files" msgstr "" -#: cps/admin.py:1532 +#: cps/admin.py:1544 msgid "Database connections are closed" msgstr "" -#: cps/admin.py:1533 +#: cps/admin.py:1545 msgid "Stopping server" msgstr "" -#: cps/admin.py:1534 +#: cps/admin.py:1546 msgid "Update finished, please press okay and reload page" msgstr "" -#: cps/admin.py:1535 cps/admin.py:1536 cps/admin.py:1537 cps/admin.py:1538 -#: cps/admin.py:1539 cps/admin.py:1540 +#: cps/admin.py:1547 cps/admin.py:1548 cps/admin.py:1549 cps/admin.py:1550 +#: cps/admin.py:1551 cps/admin.py:1552 msgid "Update failed:" msgstr "" -#: cps/admin.py:1535 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 +#: cps/admin.py:1547 cps/updater.py:391 cps/updater.py:626 cps/updater.py:628 msgid "HTTP Error" msgstr "" -#: cps/admin.py:1536 cps/updater.py:393 cps/updater.py:630 +#: cps/admin.py:1548 cps/updater.py:393 cps/updater.py:630 msgid "Connection error" msgstr "" -#: cps/admin.py:1537 cps/updater.py:395 cps/updater.py:632 +#: cps/admin.py:1549 cps/updater.py:395 cps/updater.py:632 msgid "Timeout while establishing connection" msgstr "" -#: cps/admin.py:1538 cps/updater.py:397 cps/updater.py:634 +#: cps/admin.py:1550 cps/updater.py:397 cps/updater.py:634 msgid "General error" msgstr "" -#: cps/admin.py:1539 +#: cps/admin.py:1551 msgid "Update file could not be saved in temp dir" msgstr "" -#: cps/admin.py:1540 +#: cps/admin.py:1552 msgid "Files could not be replaced during update" msgstr "" -#: cps/admin.py:1564 +#: cps/admin.py:1576 msgid "Failed to extract at least One LDAP User" msgstr "" -#: cps/admin.py:1609 +#: cps/admin.py:1621 msgid "Failed to Create at Least One LDAP User" msgstr "" -#: cps/admin.py:1622 +#: cps/admin.py:1634 #, python-format msgid "Error: %(ldaperror)s" msgstr "" -#: cps/admin.py:1626 +#: cps/admin.py:1638 msgid "Error: No user returned in response of LDAP server" msgstr "" -#: cps/admin.py:1662 +#: cps/admin.py:1674 msgid "At Least One LDAP User Not Found in Database" msgstr "" -#: cps/admin.py:1664 +#: cps/admin.py:1676 msgid "{} User Successfully Imported" msgstr "" -#: cps/admin.py:1722 +#: cps/admin.py:1734 +msgid "Books path not valid" +msgstr "" + +#: cps/admin.py:1740 msgid "DB Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1742 +#: cps/admin.py:1768 msgid "DB is not Writeable" msgstr "" -#: cps/admin.py:1758 +#: cps/admin.py:1782 msgid "Keyfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1762 +#: cps/admin.py:1786 msgid "Certfile Location is not Valid, Please Enter Correct Path" msgstr "" -#: cps/admin.py:1840 +#: cps/admin.py:1863 msgid "Password length has to be between 1 and 40" msgstr "" -#: cps/admin.py:1894 +#: cps/admin.py:1917 msgid "Database Settings updated" msgstr "" -#: cps/admin.py:1902 +#: cps/admin.py:1925 msgid "Database Configuration" msgstr "" -#: cps/admin.py:1917 cps/web.py:1290 +#: cps/admin.py:1940 cps/web.py:1299 msgid "Oops! Please complete all fields." msgstr "" -#: cps/admin.py:1926 +#: cps/admin.py:1949 msgid "E-mail is not from valid domain" msgstr "" -#: cps/admin.py:1932 +#: cps/admin.py:1955 msgid "Add new user" msgstr "" -#: cps/admin.py:1943 +#: cps/admin.py:1966 #, python-format msgid "User '%(user)s' created" msgstr "" -#: cps/admin.py:1949 +#: cps/admin.py:1972 msgid "Oops! An account already exists for this Email. or name." msgstr "" -#: cps/admin.py:1979 +#: cps/admin.py:2002 #, python-format msgid "User '%(nick)s' deleted" msgstr "" -#: cps/admin.py:1982 +#: cps/admin.py:2005 msgid "Can't delete Guest User" msgstr "" -#: cps/admin.py:1985 +#: cps/admin.py:2008 msgid "No admin user remaining, can't delete user" msgstr "" -#: cps/admin.py:2040 cps/web.py:1469 +#: cps/admin.py:2063 cps/web.py:1484 msgid "Email can't be empty and has to be a valid Email" msgstr "" -#: cps/admin.py:2066 +#: cps/admin.py:2089 #, python-format msgid "User '%(nick)s' updated" msgstr "" +#: cps/basic.py:67 cps/search.py:50 cps/search.py:426 +#: cps/templates/basic_layout.html:23 cps/templates/book_edit.html:242 +#: cps/templates/feed.xml:34 cps/templates/index.xml:12 +#: cps/templates/layout.html:47 cps/templates/layout.html:50 +#: cps/templates/search_form.html:247 +msgid "Search" +msgstr "" + #: cps/converter.py:31 msgid "not installed" msgstr "" @@ -514,127 +532,122 @@ msgstr "" msgid "Execution permissions missing" msgstr "" -#: cps/db.py:768 cps/search.py:137 cps/web.py:747 -#, python-format -msgid "Custom Column No.%(column)d does not exist in calibre database" -msgstr "" - -#: cps/db.py:1008 cps/templates/config_edit.html:203 +#: cps/db.py:1038 cps/templates/config_edit.html:203 #: cps/templates/config_view_edit.html:62 cps/templates/email_edit.html:41 -#: cps/web.py:562 cps/web.py:596 cps/web.py:641 cps/web.py:681 cps/web.py:708 -#: cps/web.py:989 cps/web.py:1019 cps/web.py:1064 cps/web.py:1092 -#: cps/web.py:1131 +#: cps/web.py:568 cps/web.py:602 cps/web.py:647 cps/web.py:687 cps/web.py:714 +#: cps/web.py:995 cps/web.py:1025 cps/web.py:1070 cps/web.py:1098 +#: cps/web.py:1137 msgid "None" msgstr "" -#: cps/editbooks.py:108 cps/editbooks.py:929 cps/web.py:529 cps/web.py:1561 -#: cps/web.py:1606 cps/web.py:1651 -msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" -msgstr "" - -#: cps/editbooks.py:151 cps/editbooks.py:1263 -msgid "User has no rights to upload cover" -msgstr "" - -#: cps/editbooks.py:171 cps/editbooks.py:744 -msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" -msgstr "" - -#: cps/editbooks.py:213 -msgid "Metadata successfully updated" -msgstr "" - -#: cps/editbooks.py:231 -msgid "Error editing book: {}" -msgstr "" - -#: cps/editbooks.py:289 +#: cps/editbooks.py:154 #, python-format msgid "File %(file)s uploaded" msgstr "" -#: cps/editbooks.py:317 +#: cps/editbooks.py:183 msgid "Source or destination format for conversion missing" msgstr "" -#: cps/editbooks.py:325 +#: cps/editbooks.py:191 #, python-format msgid "Book successfully queued for converting to %(book_format)s" msgstr "" -#: cps/editbooks.py:329 +#: cps/editbooks.py:195 #, python-format msgid "There was an error converting this book: %(res)s" msgstr "" -#: cps/editbooks.py:662 -msgid "Uploaded book probably exists in the library, consider to change before upload new: " +#: cps/editbooks.py:433 cps/editbooks.py:928 cps/web.py:535 cps/web.py:1576 +#: cps/web.py:1622 cps/web.py:1672 +msgid "Oops! Selected book is unavailable. File does not exist or is not accessible" msgstr "" -#: cps/editbooks.py:718 cps/editbooks.py:1049 +#: cps/editbooks.py:479 cps/editbooks.py:1285 +msgid "User has no rights to upload cover" +msgstr "" + +#: cps/editbooks.py:500 cps/editbooks.py:743 +msgid "Identifiers are not Case Sensitive, Overwriting Old Identifier" +msgstr "" + +#: cps/editbooks.py:515 cps/editbooks.py:717 cps/editbooks.py:1055 #, python-format msgid "'%(langname)s' is not a valid language" msgstr "" -#: cps/editbooks.py:756 cps/editbooks.py:1192 +#: cps/editbooks.py:543 +msgid "Metadata successfully updated" +msgstr "" + +#: cps/editbooks.py:566 +msgid "Error editing book: {}" +msgstr "" + +#: cps/editbooks.py:661 +msgid "Uploaded book probably exists in the library, consider to change before upload new: " +msgstr "" + +#: cps/editbooks.py:755 cps/editbooks.py:1202 msgid "File type isn't allowed to be uploaded to this server" msgstr "" -#: cps/editbooks.py:762 cps/editbooks.py:1202 +#: cps/editbooks.py:761 cps/editbooks.py:1213 #, python-format msgid "File extension '%(ext)s' is not allowed to be uploaded to this server" msgstr "" -#: cps/editbooks.py:766 cps/editbooks.py:1206 +#: cps/editbooks.py:765 cps/editbooks.py:1218 msgid "File to be uploaded must have an extension" msgstr "" -#: cps/editbooks.py:774 +#: cps/editbooks.py:773 #, python-format msgid "File %(filename)s could not saved to temp dir" msgstr "" -#: cps/editbooks.py:794 +#: cps/editbooks.py:793 #, python-format msgid "Failed to Move Cover File %(file)s: %(error)s" msgstr "" -#: cps/editbooks.py:851 cps/editbooks.py:853 +#: cps/editbooks.py:850 cps/editbooks.py:852 msgid "Book Format Successfully Deleted" msgstr "" -#: cps/editbooks.py:860 cps/editbooks.py:862 +#: cps/editbooks.py:859 cps/editbooks.py:861 msgid "Book Successfully Deleted" msgstr "" -#: cps/editbooks.py:914 +#: cps/editbooks.py:913 msgid "You are missing permissions to delete books" msgstr "" -#: cps/editbooks.py:964 +#: cps/editbooks.py:963 msgid "edit metadata" msgstr "" -#: cps/editbooks.py:1013 +#: cps/editbooks.py:1016 #, python-format -msgid "%(seriesindex)s is not a valid number, skipping" +msgid "Seriesindex: %(seriesindex)s is not a valid number, skipping" msgstr "" -#: cps/editbooks.py:1197 +#: cps/editbooks.py:1207 msgid "User has no rights to upload additional file formats" msgstr "" -#: cps/editbooks.py:1218 +#: cps/editbooks.py:1231 #, python-format msgid "Failed to create path %(path)s (Permission denied)." msgstr "" -#: cps/editbooks.py:1224 +#: cps/editbooks.py:1238 #, python-format msgid "Failed to store file %(file)s." msgstr "" -#: cps/editbooks.py:1248 +#: cps/editbooks.py:1263 #, python-format msgid "File format %(ext)s added to %(book)s" msgstr "" @@ -647,468 +660,461 @@ msgstr "" msgid "Callback domain is not verified, please follow steps to verify domain in google developer console" msgstr "" -#: cps/helper.py:86 +#: cps/helper.py:87 #, python-format msgid "%(format)s format not found for book id: %(book)d" msgstr "" -#: cps/helper.py:93 cps/tasks/convert.py:89 +#: cps/helper.py:94 cps/tasks/convert.py:93 #, python-format msgid "%(format)s not found on Google Drive: %(fn)s" msgstr "" -#: cps/helper.py:98 +#: cps/helper.py:99 #, python-format msgid "%(format)s not found: %(fn)s" msgstr "" -#: cps/helper.py:103 cps/helper.py:228 cps/templates/detail.html:58 +#: cps/helper.py:104 cps/helper.py:233 cps/templates/detail.html:66 msgid "Send to eReader" msgstr "" -#: cps/helper.py:104 cps/helper.py:122 cps/helper.py:230 +#: cps/helper.py:105 cps/helper.py:125 cps/helper.py:235 msgid "This Email has been sent via Calibre-Web." msgstr "" -#: cps/helper.py:120 +#: cps/helper.py:123 msgid "Calibre-Web Test Email" msgstr "" -#: cps/helper.py:121 +#: cps/helper.py:124 msgid "Test Email" msgstr "" -#: cps/helper.py:138 +#: cps/helper.py:141 msgid "Get Started with Calibre-Web" msgstr "" -#: cps/helper.py:143 +#: cps/helper.py:146 #, python-format msgid "Registration Email for user: %(name)s" msgstr "" -#: cps/helper.py:154 cps/helper.py:160 +#: cps/helper.py:157 cps/helper.py:163 #, python-format msgid "Convert %(orig)s to %(format)s and send to eReader" msgstr "" -#: cps/helper.py:179 cps/helper.py:183 cps/helper.py:187 +#: cps/helper.py:182 cps/helper.py:186 cps/helper.py:190 #, python-format msgid "Send %(format)s to eReader" msgstr "" -#: cps/helper.py:227 +#: cps/helper.py:230 #, python-format msgid "%(book)s send to eReader" msgstr "" -#: cps/helper.py:232 +#: cps/helper.py:237 msgid "The requested file could not be read. Maybe wrong permissions?" msgstr "" -#: cps/helper.py:347 +#: cps/helper.py:352 msgid "Read status could not set: {}" msgstr "" -#: cps/helper.py:370 +#: cps/helper.py:375 #, python-format msgid "Deleting bookfolder for book %(id)s failed, path has subfolders: %(path)s" msgstr "" -#: cps/helper.py:376 +#: cps/helper.py:381 #, python-format msgid "Deleting book %(id)s failed: %(message)s" msgstr "" -#: cps/helper.py:387 +#: cps/helper.py:392 #, python-format msgid "Deleting book %(id)s from database only, book path in database not valid: %(path)s" msgstr "" -#: cps/helper.py:463 +#: cps/helper.py:439 #, python-format msgid "Rename author from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "" -#: cps/helper.py:534 cps/helper.py:543 +#: cps/helper.py:507 cps/helper.py:516 #, python-format msgid "File %(file)s not found on Google Drive" msgstr "" -#: cps/helper.py:584 +#: cps/helper.py:559 #, python-format msgid "Rename title from: '%(src)s' to '%(dest)s' failed with error: %(error)s" msgstr "" -#: cps/helper.py:622 +#: cps/helper.py:597 #, python-format msgid "Book path %(path)s not found on Google Drive" msgstr "" -#: cps/helper.py:682 +#: cps/helper.py:657 msgid "Found an existing account for this Email address" msgstr "" -#: cps/helper.py:690 +#: cps/helper.py:665 msgid "This username is already taken" msgstr "" -#: cps/helper.py:702 +#: cps/helper.py:679 msgid "Invalid Email address format" msgstr "" -#: cps/helper.py:723 +#: cps/helper.py:701 msgid "Password doesn't comply with password validation rules" msgstr "" -#: cps/helper.py:870 +#: cps/helper.py:847 msgid "Python module 'advocate' is not installed but is needed for cover uploads" msgstr "" -#: cps/helper.py:880 +#: cps/helper.py:857 msgid "Error Downloading Cover" msgstr "" -#: cps/helper.py:883 +#: cps/helper.py:860 msgid "Cover Format Error" msgstr "" -#: cps/helper.py:886 +#: cps/helper.py:863 msgid "You are not allowed to access localhost or the local network for cover uploads" msgstr "" -#: cps/helper.py:896 +#: cps/helper.py:873 msgid "Failed to create path for cover" msgstr "" -#: cps/helper.py:912 +#: cps/helper.py:889 msgid "Cover-file is not a valid image file, or could not be stored" msgstr "" -#: cps/helper.py:923 +#: cps/helper.py:900 msgid "Only jpg/jpeg/png/webp/bmp files are supported as coverfile" msgstr "" -#: cps/helper.py:935 +#: cps/helper.py:912 msgid "Invalid cover file content" msgstr "" -#: cps/helper.py:939 +#: cps/helper.py:916 msgid "Only jpg/jpeg files are supported as coverfile" msgstr "" -#: cps/helper.py:1011 cps/helper.py:1168 +#: cps/helper.py:989 cps/helper.py:1149 msgid "Cover" msgstr "" -#: cps/helper.py:1028 +#: cps/helper.py:1006 msgid "UnRar binary file not found" msgstr "" -#: cps/helper.py:1039 +#: cps/helper.py:1017 msgid "Error executing UnRar" msgstr "" -#: cps/helper.py:1047 +#: cps/helper.py:1025 msgid "Could not find the specified directory" msgstr "" -#: cps/helper.py:1050 +#: cps/helper.py:1028 msgid "Please specify a directory, not a file" msgstr "" -#: cps/helper.py:1064 +#: cps/helper.py:1042 msgid "Calibre binaries not viable" msgstr "" -#: cps/helper.py:1073 +#: cps/helper.py:1051 #, python-format msgid "Missing calibre binaries: %(missing)s" msgstr "" -#: cps/helper.py:1075 +#: cps/helper.py:1053 #, python-format msgid "Missing executable permissions: %(missing)s" msgstr "" -#: cps/helper.py:1080 +#: cps/helper.py:1058 msgid "Error executing Calibre" msgstr "" -#: cps/helper.py:1170 cps/templates/admin.html:216 +#: cps/helper.py:1151 cps/templates/admin.html:216 msgid "Queue all books for metadata backup" msgstr "" -#: cps/kobo_auth.py:90 +#: cps/kobo_auth.py:92 msgid "Please access Calibre-Web from non localhost to get valid api_endpoint for kobo device" msgstr "" -#: cps/kobo_auth.py:116 +#: cps/kobo_auth.py:118 msgid "Kobo Setup" msgstr "" -#: cps/oauth_bb.py:77 +#: cps/oauth_bb.py:78 #, python-format msgid "Register with %(provider)s" msgstr "" -#: cps/oauth_bb.py:138 cps/remotelogin.py:130 +#: cps/oauth_bb.py:139 cps/remotelogin.py:131 #, python-format msgid "Success! You are now logged in as: %(nickname)s" msgstr "" -#: cps/oauth_bb.py:148 +#: cps/oauth_bb.py:149 #, python-format msgid "Link to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:155 +#: cps/oauth_bb.py:156 msgid "Login failed, No User Linked With OAuth Account" msgstr "" -#: cps/oauth_bb.py:197 +#: cps/oauth_bb.py:198 #, python-format msgid "Unlink to %(oauth)s Succeeded" msgstr "" -#: cps/oauth_bb.py:202 +#: cps/oauth_bb.py:203 #, python-format msgid "Unlink to %(oauth)s Failed" msgstr "" -#: cps/oauth_bb.py:205 +#: cps/oauth_bb.py:206 #, python-format msgid "Not Linked to %(oauth)s" msgstr "" -#: cps/oauth_bb.py:262 +#: cps/oauth_bb.py:263 msgid "Failed to log in with GitHub." msgstr "" -#: cps/oauth_bb.py:268 +#: cps/oauth_bb.py:269 msgid "Failed to fetch user info from GitHub." msgstr "" -#: cps/oauth_bb.py:280 +#: cps/oauth_bb.py:281 msgid "Failed to log in with Google." msgstr "" -#: cps/oauth_bb.py:286 +#: cps/oauth_bb.py:287 msgid "Failed to fetch user info from Google." msgstr "" -#: cps/oauth_bb.py:334 +#: cps/oauth_bb.py:335 msgid "GitHub Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:337 +#: cps/oauth_bb.py:338 msgid "GitHub Oauth error: {}" msgstr "" -#: cps/oauth_bb.py:358 +#: cps/oauth_bb.py:359 msgid "Google Oauth error, please retry later." msgstr "" -#: cps/oauth_bb.py:361 +#: cps/oauth_bb.py:362 msgid "Google Oauth error: {}" msgstr "" -#: cps/opds.py:298 +#: cps/opds.py:299 msgid "{} Stars" msgstr "" -#: cps/remotelogin.py:62 cps/templates/layout.html:67 -#: cps/templates/layout.html:101 cps/templates/login.html:4 -#: cps/templates/login.html:21 cps/web.py:1353 +#: cps/remotelogin.py:63 cps/templates/layout.html:69 +#: cps/templates/layout.html:104 cps/templates/login.html:4 +#: cps/templates/login.html:21 cps/web.py:1361 msgid "Login" msgstr "" -#: cps/remotelogin.py:74 cps/remotelogin.py:108 +#: cps/remotelogin.py:75 cps/remotelogin.py:109 msgid "Token not found" msgstr "" -#: cps/remotelogin.py:83 cps/remotelogin.py:116 +#: cps/remotelogin.py:84 cps/remotelogin.py:117 msgid "Token has expired" msgstr "" -#: cps/remotelogin.py:92 +#: cps/remotelogin.py:93 msgid "Success! Please return to your device" msgstr "" -#: cps/render_template.py:42 cps/web.py:418 +#: cps/render_template.py:41 cps/web.py:424 msgid "Books" msgstr "" -#: cps/render_template.py:44 +#: cps/render_template.py:43 msgid "Show recent books" msgstr "" -#: cps/render_template.py:45 cps/templates/index.xml:27 +#: cps/render_template.py:44 cps/templates/index.xml:27 msgid "Hot Books" msgstr "" -#: cps/render_template.py:47 +#: cps/render_template.py:46 msgid "Show Hot Books" msgstr "" -#: cps/render_template.py:49 cps/render_template.py:54 +#: cps/render_template.py:48 cps/render_template.py:53 msgid "Downloaded Books" msgstr "" -#: cps/render_template.py:51 cps/render_template.py:56 +#: cps/render_template.py:50 cps/render_template.py:55 #: cps/templates/user_table.html:167 msgid "Show Downloaded Books" msgstr "" -#: cps/render_template.py:59 cps/templates/index.xml:36 cps/web.py:433 +#: cps/render_template.py:58 cps/templates/index.xml:36 cps/web.py:439 msgid "Top Rated Books" msgstr "" -#: cps/render_template.py:61 cps/templates/user_table.html:161 +#: cps/render_template.py:60 cps/templates/user_table.html:161 msgid "Show Top Rated Books" msgstr "" -#: cps/render_template.py:62 cps/templates/index.xml:63 -#: cps/templates/index.xml:67 cps/web.py:766 +#: cps/render_template.py:61 cps/templates/index.xml:63 +#: cps/templates/index.xml:67 cps/web.py:772 msgid "Read Books" msgstr "" -#: cps/render_template.py:64 +#: cps/render_template.py:63 msgid "Show Read and Unread" msgstr "" -#: cps/render_template.py:66 cps/templates/index.xml:70 -#: cps/templates/index.xml:74 cps/web.py:769 +#: cps/render_template.py:65 cps/templates/index.xml:70 +#: cps/templates/index.xml:74 cps/web.py:775 msgid "Unread Books" msgstr "" -#: cps/render_template.py:68 +#: cps/render_template.py:67 msgid "Show unread" msgstr "" -#: cps/render_template.py:69 +#: cps/render_template.py:68 msgid "Discover" msgstr "" -#: cps/render_template.py:71 cps/templates/index.xml:58 +#: cps/render_template.py:70 cps/templates/index.xml:58 #: cps/templates/user_table.html:159 cps/templates/user_table.html:162 msgid "Show Random Books" msgstr "" -#: cps/render_template.py:72 cps/templates/book_table.html:67 -#: cps/templates/index.xml:97 cps/web.py:1135 +#: cps/render_template.py:71 cps/templates/book_table.html:67 +#: cps/templates/index.xml:97 cps/web.py:1141 msgid "Categories" msgstr "" -#: cps/render_template.py:74 cps/templates/user_table.html:158 +#: cps/render_template.py:73 cps/templates/user_table.html:158 msgid "Show Category Section" msgstr "" -#: cps/render_template.py:75 cps/templates/book_edit.html:91 +#: cps/render_template.py:74 cps/templates/book_edit.html:86 #: cps/templates/book_table.html:68 cps/templates/index.xml:106 -#: cps/templates/search_form.html:70 cps/web.py:1025 cps/web.py:1037 +#: cps/templates/search_form.html:70 cps/web.py:1031 cps/web.py:1043 msgid "Series" msgstr "" -#: cps/render_template.py:77 cps/templates/user_table.html:157 +#: cps/render_template.py:76 cps/templates/user_table.html:157 msgid "Show Series Section" msgstr "" -#: cps/render_template.py:78 cps/templates/book_table.html:66 +#: cps/render_template.py:77 cps/templates/book_table.html:66 #: cps/templates/index.xml:79 msgid "Authors" msgstr "" -#: cps/render_template.py:80 cps/templates/user_table.html:160 +#: cps/render_template.py:79 cps/templates/user_table.html:160 msgid "Show Author Section" msgstr "" -#: cps/render_template.py:82 cps/templates/book_table.html:72 -#: cps/templates/index.xml:88 cps/web.py:993 +#: cps/render_template.py:81 cps/templates/book_table.html:72 +#: cps/templates/index.xml:88 cps/web.py:999 msgid "Publishers" msgstr "" -#: cps/render_template.py:84 cps/templates/user_table.html:163 +#: cps/render_template.py:83 cps/templates/user_table.html:163 msgid "Show Publisher Section" msgstr "" -#: cps/render_template.py:85 cps/templates/book_table.html:70 +#: cps/render_template.py:84 cps/templates/book_table.html:70 #: cps/templates/index.xml:115 cps/templates/search_form.html:108 -#: cps/web.py:1107 +#: cps/web.py:1113 msgid "Languages" msgstr "" -#: cps/render_template.py:88 cps/templates/user_table.html:155 +#: cps/render_template.py:87 cps/templates/user_table.html:155 msgid "Show Language Section" msgstr "" -#: cps/render_template.py:89 cps/templates/index.xml:124 +#: cps/render_template.py:88 cps/templates/index.xml:124 msgid "Ratings" msgstr "" -#: cps/render_template.py:91 cps/templates/user_table.html:164 +#: cps/render_template.py:90 cps/templates/user_table.html:164 msgid "Show Ratings Section" msgstr "" -#: cps/render_template.py:92 cps/templates/index.xml:133 +#: cps/render_template.py:91 cps/templates/index.xml:133 msgid "File formats" msgstr "" -#: cps/render_template.py:94 cps/templates/user_table.html:165 +#: cps/render_template.py:93 cps/templates/user_table.html:165 msgid "Show File Formats Section" msgstr "" -#: cps/render_template.py:96 cps/web.py:792 +#: cps/render_template.py:95 cps/web.py:798 msgid "Archived Books" msgstr "" -#: cps/render_template.py:98 cps/templates/user_table.html:166 +#: cps/render_template.py:97 cps/templates/user_table.html:166 msgid "Show Archived Books" msgstr "" -#: cps/render_template.py:101 cps/web.py:823 +#: cps/render_template.py:100 cps/web.py:829 msgid "Books List" msgstr "" -#: cps/render_template.py:103 cps/templates/user_table.html:168 +#: cps/render_template.py:102 cps/templates/user_table.html:168 msgid "Show Books List" msgstr "" -#: cps/search.py:48 cps/search.py:399 cps/templates/book_edit.html:236 -#: cps/templates/feed.xml:34 cps/templates/index.xml:12 -#: cps/templates/layout.html:46 cps/templates/layout.html:49 -#: cps/templates/search_form.html:227 -msgid "Search" -msgstr "" - -#: cps/search.py:188 +#: cps/search.py:201 msgid "Published after " msgstr "" -#: cps/search.py:195 +#: cps/search.py:208 msgid "Published before " msgstr "" -#: cps/search.py:217 +#: cps/search.py:230 #, python-format msgid "Rating <= %(rating)s" msgstr "" -#: cps/search.py:219 +#: cps/search.py:232 #, python-format msgid "Rating >= %(rating)s" msgstr "" -#: cps/search.py:221 +#: cps/search.py:234 #, python-format msgid "Read Status = '%(status)s'" msgstr "" -#: cps/search.py:324 +#: cps/search.py:351 msgid "Error on search for custom columns, please restart Calibre-Web" msgstr "" -#: cps/search.py:343 cps/search.py:375 cps/templates/layout.html:57 +#: cps/search.py:370 cps/search.py:402 cps/templates/layout.html:58 msgid "Advanced Search" msgstr "" @@ -1163,7 +1169,7 @@ msgstr "" msgid "Sorry you are not allowed to remove a book from this shelf" msgstr "" -#: cps/shelf.py:218 cps/templates/layout.html:157 +#: cps/shelf.py:218 cps/templates/layout.html:160 msgid "Create a Shelf" msgstr "" @@ -1216,45 +1222,45 @@ msgstr "" msgid "A private shelf with the name '%(title)s' already exists." msgstr "" -#: cps/shelf.py:473 +#: cps/shelf.py:481 #, python-format msgid "Shelf: '%(name)s'" msgstr "" -#: cps/shelf.py:477 +#: cps/shelf.py:487 msgid "Error opening shelf. Shelf does not exist or is not accessible" msgstr "" -#: cps/tasks_status.py:46 cps/templates/layout.html:88 +#: cps/tasks_status.py:47 cps/templates/layout.html:91 #: cps/templates/tasks.html:7 msgid "Tasks" msgstr "" -#: cps/tasks_status.py:62 +#: cps/tasks_status.py:63 msgid "Waiting" msgstr "" -#: cps/tasks_status.py:64 +#: cps/tasks_status.py:65 msgid "Failed" msgstr "" -#: cps/tasks_status.py:66 +#: cps/tasks_status.py:67 msgid "Started" msgstr "" -#: cps/tasks_status.py:68 +#: cps/tasks_status.py:69 msgid "Finished" msgstr "" -#: cps/tasks_status.py:70 +#: cps/tasks_status.py:71 msgid "Ended" msgstr "" -#: cps/tasks_status.py:72 +#: cps/tasks_status.py:73 msgid "Cancelled" msgstr "" -#: cps/tasks_status.py:74 +#: cps/tasks_status.py:75 msgid "Unknown Status" msgstr "" @@ -1287,169 +1293,169 @@ msgstr "" msgid "No release information available" msgstr "" -#: cps/templates/index.html:6 cps/web.py:445 +#: cps/templates/index.html:6 cps/web.py:451 msgid "Discover (Random Books)" msgstr "" -#: cps/web.py:481 +#: cps/web.py:487 msgid "Hot Books (Most Downloaded)" msgstr "" -#: cps/web.py:512 +#: cps/web.py:518 #, python-format msgid "Downloaded books by %(user)s" msgstr "" -#: cps/web.py:545 +#: cps/web.py:551 #, python-format msgid "Author: %(name)s" msgstr "" -#: cps/web.py:581 +#: cps/web.py:587 #, python-format msgid "Publisher: %(name)s" msgstr "" -#: cps/web.py:609 +#: cps/web.py:615 #, python-format msgid "Series: %(serie)s" msgstr "" -#: cps/web.py:623 +#: cps/web.py:629 msgid "Rating: None" msgstr "" -#: cps/web.py:632 +#: cps/web.py:638 #, python-format msgid "Rating: %(rating)s stars" msgstr "" -#: cps/web.py:663 +#: cps/web.py:669 #, python-format msgid "File format: %(format)s" msgstr "" -#: cps/web.py:698 +#: cps/web.py:704 #, python-format msgid "Category: %(name)s" msgstr "" -#: cps/web.py:727 +#: cps/web.py:733 #, python-format msgid "Language: %(name)s" msgstr "" -#: cps/templates/admin.html:16 cps/web.py:965 +#: cps/templates/admin.html:16 cps/web.py:971 msgid "Downloads" msgstr "" -#: cps/web.py:1067 +#: cps/web.py:1073 msgid "Ratings list" msgstr "" -#: cps/web.py:1094 +#: cps/web.py:1100 msgid "File formats list" msgstr "" -#: cps/web.py:1249 +#: cps/web.py:1259 msgid "Please configure the SMTP mail settings first..." msgstr "" -#: cps/web.py:1256 +#: cps/web.py:1265 #, python-format msgid "Success! Book queued for sending to %(eReadermail)s" msgstr "" -#: cps/web.py:1259 +#: cps/web.py:1268 #, python-format msgid "Oops! There was an error sending book: %(res)s" msgstr "" -#: cps/web.py:1261 +#: cps/web.py:1270 msgid "Oops! Please update your profile with a valid eReader Email." msgstr "" -#: cps/web.py:1277 +#: cps/web.py:1286 msgid "Please wait one minute to register next user" msgstr "" -#: cps/templates/layout.html:68 cps/templates/layout.html:102 -#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1278 -#: cps/web.py:1282 cps/web.py:1287 cps/web.py:1291 cps/web.py:1297 -#: cps/web.py:1317 cps/web.py:1321 cps/web.py:1334 cps/web.py:1337 +#: cps/templates/layout.html:70 cps/templates/layout.html:105 +#: cps/templates/login.html:27 cps/templates/register.html:17 cps/web.py:1287 +#: cps/web.py:1291 cps/web.py:1296 cps/web.py:1300 cps/web.py:1306 +#: cps/web.py:1326 cps/web.py:1330 cps/web.py:1343 cps/web.py:1346 msgid "Register" msgstr "" -#: cps/web.py:1281 cps/web.py:1385 +#: cps/web.py:1290 cps/web.py:1393 msgid "Connection error to limiter backend, please contact your administrator" msgstr "" -#: cps/web.py:1286 cps/web.py:1333 +#: cps/web.py:1295 cps/web.py:1342 msgid "Oops! Email server is not configured, please contact your administrator." msgstr "" -#: cps/web.py:1319 +#: cps/web.py:1328 msgid "Oops! Your Email is not allowed." msgstr "" -#: cps/web.py:1322 +#: cps/web.py:1331 msgid "Success! Confirmation Email has been sent." msgstr "" -#: cps/web.py:1368 cps/web.py:1391 +#: cps/web.py:1376 cps/web.py:1399 msgid "Cannot activate LDAP authentication" msgstr "" -#: cps/web.py:1381 +#: cps/web.py:1389 msgid "Please wait one minute before next login" msgstr "" -#: cps/web.py:1400 +#: cps/web.py:1408 #, python-format msgid "you are now logged in as: '%(nickname)s'" msgstr "" -#: cps/web.py:1407 +#: cps/web.py:1415 #, python-format msgid "Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known" msgstr "" -#: cps/web.py:1412 +#: cps/web.py:1420 #, python-format msgid "Could not login: %(message)s" msgstr "" -#: cps/web.py:1416 cps/web.py:1441 +#: cps/web.py:1424 cps/web.py:1449 msgid "Wrong Username or Password" msgstr "" -#: cps/web.py:1423 +#: cps/web.py:1431 msgid "New Password was sent to your email address" msgstr "" -#: cps/web.py:1427 +#: cps/web.py:1435 msgid "An unknown error occurred. Please try again later." msgstr "" -#: cps/web.py:1429 +#: cps/web.py:1437 msgid "Please enter valid username to reset password" msgstr "" -#: cps/web.py:1437 +#: cps/web.py:1445 #, python-format msgid "You are now logged in as: '%(nickname)s'" msgstr "" -#: cps/web.py:1495 cps/web.py:1545 +#: cps/web.py:1510 cps/web.py:1560 #, python-format msgid "%(name)s's Profile" msgstr "" -#: cps/web.py:1511 +#: cps/web.py:1526 msgid "Success! Profile Updated" msgstr "" -#: cps/web.py:1515 +#: cps/web.py:1530 msgid "Oops! An account already exists for this Email." msgstr "" @@ -1457,54 +1463,58 @@ msgstr "" msgid "Found no valid gmail.json file with OAuth information" msgstr "" -#: cps/tasks/convert.py:108 +#: cps/tasks/clean.py:29 +msgid "Delete temp folder contents" +msgstr "" + +#: cps/tasks/convert.py:112 #, python-format msgid "%(book)s send to E-Reader" msgstr "" -#: cps/tasks/convert.py:170 +#: cps/tasks/convert.py:177 #, python-format msgid "Calibre ebook-convert %(tool)s not found" msgstr "" -#: cps/tasks/convert.py:204 +#: cps/tasks/convert.py:211 #, python-format msgid "%(format)s format not found on disk" msgstr "" -#: cps/tasks/convert.py:208 +#: cps/tasks/convert.py:215 msgid "Ebook converter failed with unknown error" msgstr "" -#: cps/tasks/convert.py:227 +#: cps/tasks/convert.py:234 #, python-format msgid "Kepubify-converter failed: %(error)s" msgstr "" -#: cps/tasks/convert.py:248 +#: cps/tasks/convert.py:255 #, python-format msgid "Converted file not found or more than one file in folder %(folder)s" msgstr "" -#: cps/tasks/convert.py:291 -#, python-format -msgid "Ebook-converter failed: %(error)s" -msgstr "" - -#: cps/tasks/convert.py:314 +#: cps/tasks/convert.py:289 cps/tasks/convert.py:340 #, python-format msgid "Calibre failed with error: %(error)s" msgstr "" -#: cps/tasks/convert.py:319 +#: cps/tasks/convert.py:317 +#, python-format +msgid "Ebook-converter failed: %(error)s" +msgstr "" + +#: cps/tasks/convert.py:345 msgid "Convert" msgstr "" -#: cps/tasks/database.py:28 +#: cps/tasks/database.py:26 msgid "Reconnecting Calibre database" msgstr "" -#: cps/tasks/mail.py:278 +#: cps/tasks/mail.py:283 msgid "E-mail" msgstr "" @@ -1512,30 +1522,26 @@ msgstr "" msgid "Backing up Metadata" msgstr "" -#: cps/tasks/tempFolder.py:28 -msgid "Delete temp folder contents" -msgstr "" - #: cps/tasks/thumbnail.py:97 #, python-format msgid "Generated %(count)s cover thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:230 cps/tasks/thumbnail.py:443 -#: cps/tasks/thumbnail.py:511 +#: cps/tasks/thumbnail.py:233 cps/tasks/thumbnail.py:448 +#: cps/tasks/thumbnail.py:518 msgid "Cover Thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:289 +#: cps/tasks/thumbnail.py:294 msgid "Generated {0} series thumbnails" msgstr "" -#: cps/tasks/thumbnail.py:454 +#: cps/tasks/thumbnail.py:459 msgid "Clearing cover thumbnail cache" msgstr "" #: cps/tasks/upload.py:39 cps/templates/admin.html:20 -#: cps/templates/layout.html:81 cps/templates/user_table.html:145 +#: cps/templates/layout.html:83 cps/templates/user_table.html:145 msgid "Upload" msgstr "" @@ -1554,11 +1560,11 @@ msgstr "" msgid "Email" msgstr "" -#: cps/templates/admin.html:15 cps/templates/user_edit.html:28 +#: cps/templates/admin.html:15 msgid "Send to eReader Email" msgstr "" -#: cps/templates/admin.html:17 cps/templates/layout.html:91 +#: cps/templates/admin.html:17 cps/templates/layout.html:94 #: cps/templates/user_table.html:143 msgid "Admin" msgstr "" @@ -1568,8 +1574,8 @@ msgstr "" msgid "Password" msgstr "" -#: cps/templates/admin.html:22 cps/templates/detail.html:20 -#: cps/templates/detail.html:33 cps/templates/shelf.html:8 +#: cps/templates/admin.html:22 cps/templates/detail.html:28 +#: cps/templates/detail.html:41 cps/templates/shelf.html:8 #: cps/templates/user_table.html:146 msgid "Download" msgstr "" @@ -1779,13 +1785,13 @@ msgid "OK" msgstr "" #: cps/templates/admin.html:259 cps/templates/admin.html:273 -#: cps/templates/book_edit.html:214 cps/templates/book_table.html:127 +#: cps/templates/book_edit.html:220 cps/templates/book_table.html:127 #: cps/templates/config_db.html:66 cps/templates/config_edit.html:427 -#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:342 +#: cps/templates/config_view_edit.html:175 cps/templates/detail.html:350 #: cps/templates/modal_dialogs.html:64 cps/templates/modal_dialogs.html:99 #: cps/templates/modal_dialogs.html:117 cps/templates/modal_dialogs.html:135 #: cps/templates/schedule_edit.html:45 cps/templates/shelf_edit.html:27 -#: cps/templates/tasks.html:46 cps/templates/user_edit.html:144 +#: cps/templates/tasks.html:47 cps/templates/user_edit.html:144 msgid "Cancel" msgstr "" @@ -1835,16 +1841,76 @@ msgstr "" msgid "Sort according to publishing date, oldest first" msgstr "" -#: cps/templates/author.html:56 cps/templates/author.html:115 +#: cps/templates/author.html:56 cps/templates/author.html:113 #: cps/templates/index.html:30 cps/templates/index.html:113 -#: cps/templates/search.html:67 cps/templates/shelf.html:55 +#: cps/templates/search.html:67 cps/templates/shelf.html:58 msgid "reduce" msgstr "" -#: cps/templates/author.html:99 +#: cps/templates/author.html:97 msgid "More by" msgstr "" +#: cps/templates/basic_detail.html:34 cps/templates/detail.html:158 +#: cps/templates/listenmp3.html:62 +#, python-format +msgid "Book %(index)s of %(range)s" +msgstr "" + +#: cps/templates/basic_detail.html:41 cps/templates/book_edit.html:106 +#: cps/templates/detail.html:165 cps/templates/listenmp3.html:69 +#: cps/templates/user_edit.html:33 +msgid "Language" +msgstr "" + +#: cps/templates/basic_detail.html:61 cps/templates/book_edit.html:102 +#: cps/templates/book_edit.html:279 cps/templates/book_edit.html:296 +#: cps/templates/detail.html:200 cps/templates/listenmp3.html:102 +#: cps/templates/search_form.html:16 +msgid "Publisher" +msgstr "" + +#: cps/templates/basic_detail.html:70 cps/templates/detail.html:209 +#: cps/templates/listenmp3.html:111 +msgid "Published" +msgstr "" + +#: cps/templates/basic_detail.html:76 cps/templates/detail.html:286 +#: cps/templates/listenmp3.html:177 +msgid "Description:" +msgstr "" + +#: cps/templates/basic_index.html:7 cps/templates/layout.html:175 +msgid "Previous" +msgstr "" + +#: cps/templates/basic_index.html:12 cps/templates/feed.xml:22 +#: cps/templates/layout.html:190 +msgid "Next" +msgstr "" + +#: cps/templates/basic_index.html:18 cps/templates/search.html:6 +msgid "No Results Found" +msgstr "" + +#: cps/templates/basic_layout.html:17 cps/templates/layout.html:26 +#: cps/templates/login.html:30 +msgid "Home" +msgstr "" + +#: cps/templates/basic_layout.html:21 cps/templates/layout.html:48 +msgid "Search Library" +msgstr "" + +#: cps/templates/basic_layout.html:29 cps/templates/layout.html:73 +#: cps/templates/layout.html:99 +msgid "Logout" +msgstr "" + +#: cps/templates/basic_layout.html:35 +msgid "Normal Theme" +msgstr "" + #: cps/templates/book_edit.html:11 msgid "Delete Book" msgstr "" @@ -1873,99 +1939,107 @@ msgstr "" msgid "Convert book" msgstr "" -#: cps/templates/book_edit.html:56 cps/templates/search_form.html:8 +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +#: cps/templates/layout.html:137 +msgid "Uploading..." +msgstr "" + +#: cps/templates/book_edit.html:53 cps/templates/book_edit.html:257 +#: cps/templates/layout.html:80 cps/templates/layout.html:206 +#: cps/templates/modal_dialogs.html:34 cps/templates/user_edit.html:163 +msgid "Close" +msgstr "" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Error" +msgstr "" + +#: cps/templates/book_edit.html:53 cps/templates/layout.html:80 +msgid "Upload done, processing, please wait..." +msgstr "" + +#: cps/templates/book_edit.html:58 +msgid "Upload Format" +msgstr "" + +#: cps/templates/book_edit.html:71 cps/templates/search_form.html:8 msgid "Book Title" msgstr "" -#: cps/templates/book_edit.html:63 cps/templates/book_edit.html:271 -#: cps/templates/book_edit.html:289 cps/templates/search_form.html:12 +#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:277 +#: cps/templates/book_edit.html:295 cps/templates/search_form.html:12 msgid "Author" msgstr "" -#: cps/templates/book_edit.html:68 cps/templates/book_edit.html:276 -#: cps/templates/book_edit.html:291 cps/templates/search_form.html:154 +#: cps/templates/book_edit.html:82 cps/templates/search_form.html:52 +msgid "Tags" +msgstr "" + +#: cps/templates/book_edit.html:90 +msgid "Series ID" +msgstr "" + +#: cps/templates/book_edit.html:93 +msgid "Published Date" +msgstr "" + +#: cps/templates/book_edit.html:110 +msgid "Rating" +msgstr "" + +#: cps/templates/book_edit.html:114 cps/templates/book_edit.html:282 +#: cps/templates/book_edit.html:297 cps/templates/search_form.html:154 msgid "Description" msgstr "" -#: cps/templates/book_edit.html:73 +#: cps/templates/book_edit.html:118 msgid "Identifiers" msgstr "" -#: cps/templates/book_edit.html:77 cps/templates/book_edit.html:300 +#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:306 msgid "Identifier Type" msgstr "" -#: cps/templates/book_edit.html:78 cps/templates/book_edit.html:301 +#: cps/templates/book_edit.html:123 cps/templates/book_edit.html:307 msgid "Identifier Value" msgstr "" -#: cps/templates/book_edit.html:79 cps/templates/book_edit.html:302 +#: cps/templates/book_edit.html:124 cps/templates/book_edit.html:308 #: cps/templates/user_table.html:24 msgid "Remove" msgstr "" -#: cps/templates/book_edit.html:83 +#: cps/templates/book_edit.html:129 msgid "Add Identifier" msgstr "" -#: cps/templates/book_edit.html:87 cps/templates/search_form.html:52 -msgid "Tags" -msgstr "" - -#: cps/templates/book_edit.html:95 -msgid "Series ID" -msgstr "" - -#: cps/templates/book_edit.html:99 -msgid "Rating" -msgstr "" - -#: cps/templates/book_edit.html:104 +#: cps/templates/book_edit.html:133 msgid "Fetch Cover from URL (JPEG - Image will be downloaded and stored in database)" msgstr "" -#: cps/templates/book_edit.html:108 +#: cps/templates/book_edit.html:137 msgid "Upload Cover from Local Disk" msgstr "" -#: cps/templates/book_edit.html:113 -msgid "Published Date" -msgstr "" - -#: cps/templates/book_edit.html:122 cps/templates/book_edit.html:273 -#: cps/templates/book_edit.html:290 cps/templates/detail.html:192 -#: cps/templates/listenmp3.html:102 cps/templates/search_form.html:16 -msgid "Publisher" -msgstr "" - -#: cps/templates/book_edit.html:126 cps/templates/detail.html:157 -#: cps/templates/listenmp3.html:69 cps/templates/user_edit.html:33 -msgid "Language" -msgstr "" - -#: cps/templates/book_edit.html:136 cps/templates/search_form.html:46 -#: cps/templates/search_form.html:165 +#: cps/templates/book_edit.html:149 cps/templates/search_form.html:46 +#: cps/templates/search_form.html:167 msgid "Yes" msgstr "" -#: cps/templates/book_edit.html:137 cps/templates/search_form.html:47 -#: cps/templates/search_form.html:166 +#: cps/templates/book_edit.html:150 cps/templates/search_form.html:47 +#: cps/templates/search_form.html:168 msgid "No" msgstr "" -#: cps/templates/book_edit.html:201 -msgid "Upload Format" -msgstr "" - -#: cps/templates/book_edit.html:209 +#: cps/templates/book_edit.html:215 msgid "View Book on Save" msgstr "" -#: cps/templates/book_edit.html:212 cps/templates/book_edit.html:230 +#: cps/templates/book_edit.html:218 cps/templates/book_edit.html:236 msgid "Fetch Metadata" msgstr "" -#: cps/templates/book_edit.html:213 cps/templates/config_db.html:65 +#: cps/templates/book_edit.html:219 cps/templates/config_db.html:65 #: cps/templates/config_edit.html:426 cps/templates/config_view_edit.html:174 #: cps/templates/email_edit.html:65 cps/templates/schedule_edit.html:44 #: cps/templates/shelf_edit.html:25 cps/templates/shelf_order.html:41 @@ -1973,37 +2047,31 @@ msgstr "" msgid "Save" msgstr "" -#: cps/templates/book_edit.html:233 +#: cps/templates/book_edit.html:239 msgid "Keyword" msgstr "" -#: cps/templates/book_edit.html:234 +#: cps/templates/book_edit.html:240 msgid "Search keyword" msgstr "" -#: cps/templates/book_edit.html:240 +#: cps/templates/book_edit.html:246 msgid "Click the cover to load metadata to the form" msgstr "" -#: cps/templates/book_edit.html:247 cps/templates/book_edit.html:286 +#: cps/templates/book_edit.html:253 cps/templates/book_edit.html:292 msgid "Loading..." msgstr "" -#: cps/templates/book_edit.html:251 cps/templates/layout.html:78 -#: cps/templates/layout.html:203 cps/templates/modal_dialogs.html:34 -#: cps/templates/user_edit.html:163 -msgid "Close" -msgstr "" - -#: cps/templates/book_edit.html:278 cps/templates/book_edit.html:292 +#: cps/templates/book_edit.html:284 cps/templates/book_edit.html:298 msgid "Source" msgstr "" -#: cps/templates/book_edit.html:287 +#: cps/templates/book_edit.html:293 msgid "Search error!" msgstr "" -#: cps/templates/book_edit.html:288 +#: cps/templates/book_edit.html:294 msgid "No Result(s) found! Please try another keyword." msgstr "" @@ -2110,7 +2178,7 @@ msgid "Enter " msgstr "" #: cps/templates/book_table.html:113 cps/templates/modal_dialogs.html:46 -#: cps/templates/tasks.html:36 +#: cps/templates/tasks.html:37 msgid "Are you really sure?" msgstr "" @@ -2582,72 +2650,59 @@ msgstr "" msgid "Add Allowed/Denied custom column values" msgstr "" -#: cps/templates/detail.html:77 cps/templates/detail.html:91 +#: cps/templates/detail.html:85 cps/templates/detail.html:99 msgid "Read in Browser" msgstr "" -#: cps/templates/detail.html:100 cps/templates/detail.html:120 +#: cps/templates/detail.html:108 cps/templates/detail.html:128 msgid "Listen in Browser" msgstr "" -#: cps/templates/detail.html:150 cps/templates/listenmp3.html:62 -#, python-format -msgid "Book %(index)s of %(range)s" -msgstr "" - -#: cps/templates/detail.html:201 cps/templates/listenmp3.html:111 -msgid "Published" -msgstr "" - -#: cps/templates/detail.html:251 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:259 cps/templates/listenmp3.html:158 msgid "Mark As Unread" msgstr "" -#: cps/templates/detail.html:252 cps/templates/listenmp3.html:158 +#: cps/templates/detail.html:260 cps/templates/listenmp3.html:158 msgid "Mark As Read" msgstr "" -#: cps/templates/detail.html:254 +#: cps/templates/detail.html:262 msgid "Mark Book as Read or Unread" msgstr "" -#: cps/templates/detail.html:254 cps/templates/listenmp3.html:159 +#: cps/templates/detail.html:262 cps/templates/listenmp3.html:159 msgid "Read" msgstr "" -#: cps/templates/detail.html:264 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:272 cps/templates/listenmp3.html:166 msgid "Restore from archive" msgstr "" -#: cps/templates/detail.html:265 cps/templates/listenmp3.html:166 +#: cps/templates/detail.html:273 cps/templates/listenmp3.html:166 msgid "Add to archive" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Mark Book as archived or not, to hide it in Calibre-Web and delete it from Kobo Reader" msgstr "" -#: cps/templates/detail.html:267 +#: cps/templates/detail.html:275 msgid "Archive" msgstr "" -#: cps/templates/detail.html:278 cps/templates/listenmp3.html:177 -msgid "Description:" -msgstr "" - -#: cps/templates/detail.html:293 cps/templates/listenmp3.html:190 +#: cps/templates/detail.html:301 cps/templates/listenmp3.html:190 #: cps/templates/search.html:16 msgid "Add to shelf" msgstr "" -#: cps/templates/detail.html:305 cps/templates/detail.html:324 -#: cps/templates/feed.xml:81 cps/templates/layout.html:154 +#: cps/templates/detail.html:313 cps/templates/detail.html:332 +#: cps/templates/feed.xml:81 cps/templates/layout.html:157 #: cps/templates/listenmp3.html:201 cps/templates/listenmp3.html:218 #: cps/templates/search.html:22 msgid "(Public)" msgstr "" -#: cps/templates/detail.html:340 +#: cps/templates/detail.html:348 msgid "Edit Metadata" msgstr "" @@ -2717,10 +2772,6 @@ msgstr "" msgid "Denied Domains (Blacklist)" msgstr "" -#: cps/templates/feed.xml:22 cps/templates/layout.html:187 -msgid "Next" -msgstr "" - #: cps/templates/generate_kobo_auth_url.html:6 msgid "Open the .kobo/Kobo/Kobo eReader.conf file in a text editor and add (or edit):" msgstr "" @@ -2741,11 +2792,15 @@ msgstr "" msgid "Create Issue" msgstr "" -#: cps/templates/http_error.html:51 +#: cps/templates/http_error.html:52 +msgid "Return to Database config" +msgstr "" + +#: cps/templates/http_error.html:54 msgid "Return to Home" msgstr "" -#: cps/templates/http_error.html:53 +#: cps/templates/http_error.html:57 msgid "Logout User" msgstr "" @@ -2835,7 +2890,7 @@ msgstr "" msgid "Books ordered by file formats" msgstr "" -#: cps/templates/index.xml:142 cps/templates/layout.html:152 +#: cps/templates/index.xml:142 cps/templates/layout.html:155 #: cps/templates/search_form.html:88 msgid "Shelves" msgstr "" @@ -2844,60 +2899,36 @@ msgstr "" msgid "Books organized in shelves" msgstr "" -#: cps/templates/layout.html:26 cps/templates/login.html:30 -msgid "Home" -msgstr "" - #: cps/templates/layout.html:32 msgid "Toggle Navigation" msgstr "" -#: cps/templates/layout.html:47 -msgid "Search Library" +#: cps/templates/layout.html:59 +msgid "Simple Theme" msgstr "" -#: cps/templates/layout.html:65 cps/templates/layout.html:94 +#: cps/templates/layout.html:67 cps/templates/layout.html:97 msgid "Account" msgstr "" -#: cps/templates/layout.html:71 cps/templates/layout.html:96 -msgid "Logout" -msgstr "" - -#: cps/templates/layout.html:78 cps/templates/layout.html:134 -msgid "Uploading..." -msgstr "" - -#: cps/templates/layout.html:78 -msgid "Error" -msgstr "" - -#: cps/templates/layout.html:78 -msgid "Upload done, processing, please wait..." -msgstr "" - -#: cps/templates/layout.html:91 cps/templates/read.html:76 +#: cps/templates/layout.html:94 cps/templates/read.html:78 #: cps/templates/readcbr.html:70 cps/templates/readcbr.html:96 msgid "Settings" msgstr "" -#: cps/templates/layout.html:135 +#: cps/templates/layout.html:138 msgid "Please do not refresh the page" msgstr "" -#: cps/templates/layout.html:145 +#: cps/templates/layout.html:148 msgid "Browse" msgstr "" -#: cps/templates/layout.html:158 cps/templates/stats.html:3 +#: cps/templates/layout.html:161 cps/templates/stats.html:3 msgid "About" msgstr "" -#: cps/templates/layout.html:172 -msgid "Previous" -msgstr "" - -#: cps/templates/layout.html:199 +#: cps/templates/layout.html:202 msgid "Book Details" msgstr "" @@ -3013,7 +3044,7 @@ msgstr "" msgid "Select" msgstr "" -#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:45 +#: cps/templates/modal_dialogs.html:134 cps/templates/tasks.html:46 msgid "Ok" msgstr "" @@ -3021,34 +3052,74 @@ msgstr "" msgid "Calibre-Web eBook Catalog" msgstr "" -#: cps/templates/read.html:6 +#: cps/templates/read.html:7 msgid "epub Reader" msgstr "" -#: cps/templates/read.html:81 cps/templates/readcbr.html:104 +#: cps/templates/read.html:80 +msgid "Choose a theme below:" +msgstr "" + +#: cps/templates/read.html:84 cps/templates/readcbr.html:104 msgid "Light" msgstr "" -#: cps/templates/read.html:82 cps/templates/readcbr.html:105 +#: cps/templates/read.html:86 cps/templates/readcbr.html:105 msgid "Dark" msgstr "" -#: cps/templates/read.html:83 +#: cps/templates/read.html:88 msgid "Sepia" msgstr "" -#: cps/templates/read.html:84 +#: cps/templates/read.html:90 msgid "Black" msgstr "" -#: cps/templates/read.html:88 +#: cps/templates/read.html:95 msgid "Reflow text when sidebars are open." msgstr "" -#: cps/templates/read.html:93 +#: cps/templates/read.html:100 msgid "Font Sizes" msgstr "" +#: cps/templates/read.html:105 +msgid "Font" +msgstr "" + +#: cps/templates/read.html:106 +msgid "Default" +msgstr "" + +#: cps/templates/read.html:107 +msgid "Yahei" +msgstr "" + +#: cps/templates/read.html:108 +msgid "SimSun" +msgstr "" + +#: cps/templates/read.html:109 +msgid "KaiTi" +msgstr "" + +#: cps/templates/read.html:110 +msgid "Arial" +msgstr "" + +#: cps/templates/read.html:113 +msgid "Spread" +msgstr "" + +#: cps/templates/read.html:114 +msgid "Two columns" +msgstr "" + +#: cps/templates/read.html:115 +msgid "One column" +msgstr "" + #: cps/templates/readcbr.html:8 msgid "Comic Reader" msgstr "" @@ -3225,10 +3296,6 @@ msgstr "" msgid "Generate Series Cover Thumbnails" msgstr "" -#: cps/templates/search.html:6 -msgid "No Results Found" -msgstr "" - #: cps/templates/search.html:7 msgid "Search Term:" msgstr "" @@ -3245,11 +3312,11 @@ msgstr "" msgid "Published Date To" msgstr "" -#: cps/templates/search_form.html:44 +#: cps/templates/search_form.html:44 cps/templates/search_form.html:165 msgid "Any" msgstr "" -#: cps/templates/search_form.html:45 +#: cps/templates/search_form.html:45 cps/templates/search_form.html:166 msgid "Empty" msgstr "" @@ -3285,11 +3352,13 @@ msgstr "" msgid "Rating Below" msgstr "" -#: cps/templates/search_form.html:181 +#: cps/templates/search_form.html:175 cps/templates/search_form.html:187 +#: cps/templates/search_form.html:201 msgid "From:" msgstr "" -#: cps/templates/search_form.html:191 +#: cps/templates/search_form.html:179 cps/templates/search_form.html:191 +#: cps/templates/search_form.html:211 msgid "To:" msgstr "" @@ -3313,6 +3382,14 @@ msgstr "" msgid "Enable Change order" msgstr "" +#: cps/templates/shelf.html:28 +msgid "Sort according to book added to shelf, newest first" +msgstr "" + +#: cps/templates/shelf.html:29 +msgid "Sort according to book added to shelf, oldest first" +msgstr "" + #: cps/templates/shelf_edit.html:14 msgid "Share with Everyone" msgstr "" @@ -3381,15 +3458,19 @@ msgstr "" msgid "Run Time" msgstr "" -#: cps/templates/tasks.html:20 +#: cps/templates/tasks.html:19 +msgid "Message" +msgstr "" + +#: cps/templates/tasks.html:21 msgid "Actions" msgstr "" -#: cps/templates/tasks.html:40 +#: cps/templates/tasks.html:41 msgid "This task will be cancelled. Any progress made by this task will be saved." msgstr "" -#: cps/templates/tasks.html:41 +#: cps/templates/tasks.html:42 msgid "If this is a scheduled task, it will be re-ran during the next scheduled time." msgstr "" @@ -3397,6 +3478,10 @@ msgstr "" msgid "Reset user Password" msgstr "" +#: cps/templates/user_edit.html:28 +msgid "Send to eReader Email Address. Use comma to separate emails for multiple eReaders" +msgstr "" + #: cps/templates/user_edit.html:43 msgid "Language of Books" msgstr "" diff --git a/optional-requirements.txt b/optional-requirements.txt index d63bc41c..027389d7 100644 --- a/optional-requirements.txt +++ b/optional-requirements.txt @@ -1,13 +1,13 @@ # GDrive Integration google-api-python-client>=1.7.11,<2.200.0 -gevent>20.6.0,<24.3.0 -greenlet>=0.4.17,<3.1.0 +gevent>20.6.0,<24.12.0 +greenlet>=0.4.17,<3.2.0 httplib2>=0.9.2,<0.23.0 oauth2client>=4.0.0,<4.1.4 uritemplate>=3.0.0,<4.2.0 -pyasn1-modules>=0.0.8,<0.5.0 +pyasn1-modules>=0.0.8,<0.7.0 pyasn1>=0.1.9,<0.7.0 -PyDrive2>=1.3.1,<1.20.0 +PyDrive2>=1.3.1,<1.22.0 PyYAML>=3.12,<6.1 rsa>=3.4.2,<4.10.0 @@ -17,25 +17,27 @@ google-api-python-client>=1.7.11,<2.200.0 # goodreads goodreads>=0.3.2,<0.4.0 -python-Levenshtein>=0.12.0,<0.26.0 +python-Levenshtein>=0.12.0,<0.28.0 # ldap login python-ldap>=3.0.0,<3.5.0 Flask-SimpleLDAP>=1.4.0,<2.1.0 # oauth -Flask-Dance>=2.0.0,<7.1.0 +Flask-Dance>=2.0.0,<7.2.0 SQLAlchemy-Utils>=0.33.5,<0.42.0 # metadata extraction rarfile>=3.2,<5.0 scholarly>=1.2.0,<1.8 -markdown2>=2.0.0,<2.5.0 -html2text>=2020.1.16,<2024.2.26 +markdown2>=2.0.0,<2.6.0 +html2text>=2020.1.16,<2025.2.26 python-dateutil>=2.1,<2.10.0 -beautifulsoup4>=4.0.1,<4.13.0 +beautifulsoup4>=4.0.1,<4.14.0 faust-cchardet>=2.1.18,<2.1.20 py7zr>=0.15.0,<0.21.0 +mutagen>=1.40.0,<1.50.0 +pycountry>=20.0.0,<25.0.0 # Comics natsort>=2.2.0,<8.5.0 diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 00000000..778e65e9 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,129 @@ +[build-system] +requires = ["setuptools>=61.2"] +build-backend = "setuptools.build_meta" + +[project] +name = "calibreweb" +description = "Web app for browsing, reading and downloading eBooks stored in a Calibre database." +authors = [{name = "@OzzieIsaacs", email = "Ozzie.Fernandez.Isaacs@googlemail.com"}] +maintainers = [{name = "@OzzieIsaacs"}] +license = {text = "GPLv3+"} +classifiers = [ + "Development Status :: 5 - Production/Stable", + "License :: OSI Approved :: GNU Affero General Public License v3", + "Programming Language :: Python :: 3", + "Programming Language :: Python :: 3.8", + "Programming Language :: Python :: 3.9", + "Programming Language :: Python :: 3.10", + "Programming Language :: Python :: 3.11", + "Programming Language :: Python :: 3.12", + "Programming Language :: Python :: 3.13", + "Operating System :: OS Independent", +] +keywords = [ + "calibre", + "calibre-web", + "library", +] +requires-python = ">=3.6" +dependencies = [ + "APScheduler>=3.6.3,<3.12.0", + "Babel>=1.3,<3.0", + "Flask-Babel>=3.0.0,<4.1.0", + "Flask-Principal>=0.3.2,<0.5.1", + "Flask>=1.0.2,<3.2.0", + "iso-639>=0.4.5,<0.5.0;python_version<'3.12'", + "pycountry>=20.0.0,<25.0.0;python_version>='3.12'", + "PyPDF>=3.15.6,<5.5.0", + "pytz>=2016.10", + "requests>=2.32.0,<2.33.0", + "SQLAlchemy>=1.3.0,<2.1.0", + "tornado>=6.4.2,<6.6", + "Wand>=0.4.4,<0.7.0", + "unidecode>=0.04.19,<1.4.0", + "lxml>=4.9.1,<5.4.0", + "flask-wtf>=0.14.2,<1.3.0", + "chardet>=3.0.0,<5.3.0", + "netifaces-plus>=0.12.0,<0.13.0", + "urllib3>=1.22,<3.0", + "Flask-Limiter>=2.3.0,<3.13.0", + "regex>=2022.3.2,<2025.3.20", + "bleach>=6.0.0,<6.2.0", + "python-magic>=0.4.27,<0.5.0", + "python-magic-bin>=0.4.0,<0.5.0;sys_platform=='win32'", + "flask-httpAuth>=4.4.0,<5.0.0", + "cryptography>=43.0.4,<45.0.0", +] +dynamic = ["version"] + +[project.urls] +Homepage = "https://github.com/janeczku/calibre-web" +"Bug Tracker" = "https://github.com/janeczku/calibre-web/issues" +"Release Management" = "https://github.com/janeczku/calibre-web/releases" +Documentation = "https://github.com/janeczku/calibre-web/wiki" +"Source Code" = "https://github.com/janeczku/calibre-web" + +[project.readme] +file = "README.md" +content-type = "text/markdown" + +[project.optional-dependencies] +gdrive = [ + "google-api-python-client>=1.7.11,<2.200.0", + "gevent>20.6.0,<24.12.0", + "greenlet>=0.4.17,<3.2.0", + "httplib2>=0.9.2,<0.23.0", + "oauth2client>=4.0.0,<4.1.4", + "uritemplate>=3.0.0,<4.2.0", + "pyasn1-modules>=0.0.8,<0.7.0", + "pyasn1>=0.1.9,<0.7.0", + "PyDrive2>=1.3.1,<1.22.0", + "PyYAML>=3.12,<6.1", + "rsa>=3.4.2,<4.10.0", +] +gmail = [ + "google-auth-oauthlib>=0.4.3,<1.3.0", + "google-api-python-client>=1.7.11,<2.200.0", +] +goodreads = [ + "goodreads>=0.3.2,<0.4.0", + "python-Levenshtein>=0.12.0,<0.28.0", +] +ldap = [ + "python-ldap>=3.0.0,<3.5.0", + "Flask-SimpleLDAP>=1.4.0,<2.1.0", +] +oauth = [ + "Flask-Dance>=2.0.0,<7.2.0", + "SQLAlchemy-Utils>=0.33.5,<0.42.0", +] +metadata = [ + "rarfile>=3.2,<5.0", + "scholarly>=1.2.0,<1.8", + "markdown2>=2.0.0,<2.6.0", + "html2text>=2020.1.16,<2025.2.26", + "python-dateutil>=2.1,<2.10.0", + "beautifulsoup4>=4.0.1,<4.14.0", + "faust-cchardet>=2.1.18,<2.1.20", + "py7zr>=0.15.0,<0.21.0", + "mutagen>=1.40.0,<1.50.0", + "pycountry>=20.0.0,<25.0.0", +] +comics = [ + "natsort>=2.2.0,<8.5.0", + "comicapi>=2.2.0,<3.3.0", +] +kobo = [ + "jsonschema>=3.2.0,<4.24.0", +] + +[project.scripts] +cps = "calibreweb:main" + +[tool.setuptools] +include-package-data = true +license-files = ["LICENSE"] + +[tool.setuptools.dynamic] +version = {attr = "calibreweb.cps.constants.STABLE_VERSION"} + diff --git a/requirements.txt b/requirements.txt index 460a7357..e6ee645a 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,22 +1,26 @@ -APScheduler>=3.6.3,<3.11.0 +APScheduler>=3.6.3,<3.12.0 Babel>=1.3,<3.0 -Flask-Babel>=0.11.1,<4.1.0 +Flask-Babel>=3.0.0,<4.1.0 Flask-Principal>=0.3.2,<0.5.1 -Flask>=1.0.2,<3.1.0 -iso-639>=0.4.5,<0.5.0 -PyPDF>=3.15.6,<4.3.0 +Flask>=1.0.2,<3.2.0 +iso-639>=0.4.5,<0.5.0;python_version<'3.12' +pycountry>=20.0.0,<25.0.0;python_version>='3.12' +PyPDF>=3.15.6,<5.5.0 pytz>=2016.10 -requests>=2.28.0,<2.32.0 +requests>=2.32.0,<2.33.0 SQLAlchemy>=1.3.0,<2.1.0 -tornado>=6.3,<6.5 +tornado>=6.4.2,<6.6 Wand>=0.4.4,<0.7.0 unidecode>=0.04.19,<1.4.0 -lxml>=4.9.1,<5.3.0 +lxml>=4.9.1,<5.4.0 flask-wtf>=0.14.2,<1.3.0 -chardet>=3.0.0,<4.1.0 -advocate>=1.0.0,<1.1.0 -Flask-Limiter>=2.3.0,<3.9.0 -regex>=2022.3.2,<2024.6.25 -bleach>=6.0.0,<6.2.0 +chardet>=3.0.0,<5.3.0 +netifaces-plus>=0.12.0,<0.13.0 +urllib3>=1.22,<3.0 +Flask-Limiter>=2.3.0,<3.13.0 +regex>=2022.3.2,<2025.3.20 +bleach>=6.0.0,<6.3.0 python-magic>=0.4.27,<0.5.0 +python-magic-bin>=0.4.0,<0.5.0;sys_platform=='win32' flask-httpAuth>=4.4.0,<5.0.0 +cryptography>=39.0.0,<45.0.0 diff --git a/setup.cfg b/setup.cfg deleted file mode 100644 index 26eef25d..00000000 --- a/setup.cfg +++ /dev/null @@ -1,108 +0,0 @@ -[metadata] -name = calibreweb -url = https://github.com/janeczku/calibre-web -project_urls = - Bug Tracker = https://github.com/janeczku/calibre-web/issues - Release Management = https://github.com/janeczku/calibre-web/releases - Documentation = https://github.com/janeczku/calibre-web/wiki - Source Code = https://github.com/janeczku/calibre-web -description = Web app for browsing, reading and downloading eBooks stored in a Calibre database. -long_description = file: README.md -long_description_content_type = text/markdown -author = @OzzieIsaacs -author_email = Ozzie.Fernandez.Isaacs@googlemail.com -maintainer = @OzzieIsaacs -license = GPLv3+ -license_files = LICENSE -classifiers = - Development Status :: 5 - Production/Stable - License :: OSI Approved :: GNU Affero General Public License v3 - Programming Language :: Python :: 3 - Programming Language :: Python :: 3.6 - Programming Language :: Python :: 3.7 - Programming Language :: Python :: 3.8 - Programming Language :: Python :: 3.9 - Programming Language :: Python :: 3.10 - Programming Language :: Python :: 3.11 - Operating System :: OS Independent -keywords = - calibre - calibre-web - library -python_requires = >=3.5 - -[options.entry_points] -console_scripts = - cps = calibreweb:main - -[options] -include_package_data = True -install_requires = - APScheduler>=3.6.3,<3.11.0 - Babel>=1.3,<3.0 - Flask-Babel>=0.11.1,<4.1.0 - Flask-Principal>=0.3.2,<0.5.1 - Flask>=1.0.2,<3.1.0 - iso-639>=0.4.5,<0.5.0 - PyPDF>=3.15.6,<4.3.0 - pytz>=2016.10 - requests>=2.28.0,<2.32.0 - SQLAlchemy>=1.3.0,<2.1.0 - tornado>=6.3,<6.5 - Wand>=0.4.4,<0.7.0 - unidecode>=0.04.19,<1.4.0 - lxml>=4.9.1,<5.3.0 - flask-wtf>=0.14.2,<1.3.0 - chardet>=3.0.0,<4.1.0 - advocate>=1.0.0,<1.1.0 - Flask-Limiter>=2.3.0,<3.9.0 - regex>=2022.3.2,<2024.6.25 - bleach>=6.0.0,<6.2.0 - python-magic>=0.4.27,<0.5.0 - flask-httpAuth>=4.4.0,<5.0.0 - - -[options.packages.find] -where = src -include = cps/services* - -[options.extras_require] -gdrive = - google-api-python-client>=1.7.11,<2.200.0 - gevent>20.6.0,<24.3.0 - greenlet>=0.4.17,<3.1.0 - httplib2>=0.9.2,<0.23.0 - oauth2client>=4.0.0,<4.1.4 - uritemplate>=3.0.0,<4.2.0 - pyasn1-modules>=0.0.8,<0.5.0 - pyasn1>=0.1.9,<0.7.0 - PyDrive2>=1.3.1,<1.20.0 - PyYAML>=3.12,<6.1 - rsa>=3.4.2,<4.10.0 -gmail = - google-auth-oauthlib>=0.4.3,<1.3.0 - google-api-python-client>=1.7.11,<2.200.0 -goodreads = - goodreads>=0.3.2,<0.4.0 - python-Levenshtein>=0.12.0,<0.26.0 -ldap = - python-ldap>=3.0.0,<3.5.0 - Flask-SimpleLDAP>=1.4.0,<2.1.0 -oauth = - Flask-Dance>=2.0.0,<7.1.0 - SQLAlchemy-Utils>=0.33.5,<0.42.0 -metadata = - rarfile>=3.2,<5.0 - scholarly>=1.2.0,<1.8 - markdown2>=2.0.0,<2.5.0 - html2text>=2020.1.16,<2024.2.26 - python-dateutil>=2.1,<2.10.0 - beautifulsoup4>=4.0.1,<4.13.0 - faust-cchardet>=2.1.18,<2.1.20 - py7zr>=0.15.0,<0.21.0 -comics = - natsort>=2.2.0,<8.5.0 - comicapi>=2.2.0,<3.3.0 -kobo = - jsonschema>=3.2.0,<4.24.0 - diff --git a/test/Calibre-Web TestSummary_Linux.html b/test/Calibre-Web TestSummary_Linux.html index 3bd3a93d..b6085c78 100644 --- a/test/Calibre-Web TestSummary_Linux.html +++ b/test/Calibre-Web TestSummary_Linux.html @@ -37,20 +37,20 @@
-

Start Time: 2024-08-03 20:30:20

+

Start Time: 2025-03-25 20:17:40

-

Stop Time: 2024-08-04 04:01:30

+

Stop Time: 2025-03-26 03:29:00

-

Duration: 6h 17 min

+

Duration: 6h 7 min

@@ -932,13 +932,13 @@ TestEbookConvertKepubify - 3 - 3 + 4 + 4 0 0 0 - Detail + Detail @@ -970,13 +970,22 @@ + + + +
TestEbookConvertKepubify - test_kobo_kepub_formats
+ + PASS + + + - + TestEbookConvertGDriveKepubify 3 - 2 - 1 + 3 + 0 0 0 @@ -995,33 +1004,11 @@ - +
TestEbookConvertGDriveKepubify - test_convert_only
- -
- FAIL -
- - - - + PASS @@ -1036,15 +1023,15 @@ AssertionError: 1 != 3 - + TestEditAdditionalBooks - 20 + 18 18 0 0 - 2 + 0 - Detail + Detail @@ -1185,51 +1172,16 @@ AssertionError: 1 != 3 - + -
TestEditAdditionalBooks - test_writeonly_calibre_database
- - -
- SKIP -
- - - - - - - - - - -
TestEditAdditionalBooks - test_writeonly_path
+
TestEditAdditionalBooks - test_xss_author_edit
PASS - - -
TestEditAdditionalBooks - test_xss_author_edit
- - SKIP - - - - - +
TestEditAdditionalBooks - test_xss_comment_edit
@@ -1238,7 +1190,7 @@ AssertionError: 1 != 3 - +
TestEditAdditionalBooks - test_xss_custom_comment_edit
@@ -1251,10 +1203,10 @@ AssertionError: 1 != 3 TestEditBooks 38 - 36 + 37 0 0 - 2 + 1 Detail @@ -1459,33 +1411,7 @@ AssertionError: 1 != 3 - - -
TestEditBooks - test_rename_uppercase_lowercase
- - -
- SKIP -
- - - - - - - - - +
TestEditBooks - test_typeahead_author
@@ -1494,7 +1420,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_typeahead_functions
@@ -1503,7 +1429,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_typeahead_language
@@ -1512,7 +1438,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_typeahead_publisher
@@ -1521,7 +1447,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_typeahead_series
@@ -1530,7 +1456,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_typeahead_tag
@@ -1539,7 +1465,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_cb7
@@ -1548,7 +1474,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_cbr
@@ -1557,7 +1483,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_cbt
@@ -1566,7 +1492,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_cbz
@@ -1575,7 +1501,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_epub
@@ -1584,7 +1510,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_fb2
@@ -1593,7 +1519,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_lit
@@ -1602,7 +1528,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_mobi
@@ -1611,7 +1537,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_book_pdf
@@ -1620,7 +1546,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_cbz_coverformats
@@ -1629,7 +1555,7 @@ AssertionError: 1 != 3 - +
TestEditBooks - test_upload_cover_hdd
@@ -1637,14 +1563,23 @@ AssertionError: 1 != 3 + + + +
TestEditBooks - test_xchange
+ + PASS + + + - + TestEditAuthors 9 - 5 + 9 + 0 0 - 4 0 Detail @@ -1662,118 +1597,38 @@ AssertionError: 1 != 3 - +
TestEditAuthors - test_change_capital_one_author_one_book
- -
- ERROR -
- - - - + PASS - +
TestEditAuthors - test_change_capital_one_author_two_books
- -
- ERROR -
- - - - + PASS - +
TestEditAuthors - test_change_capital_one_author_two_books_coauthor
- -
- ERROR -
- - - - + PASS - +
TestEditAuthors - test_change_capital_rename_co_author
- -
- ERROR -
- - - - + PASS @@ -2206,8 +2061,8 @@ NameError: name 'SMB_LIB' is not defined
Traceback (most recent call last):
-  File "/home/ozzie/Development/calibre-web-test/test/test_edit_books_metadata.py", line 69, in test_load_metadata
-    if 'https://comicvine.gamespot.com/' == results[10]['source']:
+  File "/home/ozzie/Development/calibre-web-test/test/test_edit_books_metadata.py", line 99, in test_load_metadata
+    if results[cont]['source'] == 'https://comicvine.gamespot.com/':
 IndexError: list index out of range
@@ -2396,13 +2251,13 @@ IndexError: list index out of range - + TestLoadMetadataScholar 1 - 0 1 0 0 + 0 Detail @@ -2410,41 +2265,21 @@ IndexError: list index out of range - +
TestLoadMetadataScholar - test_load_metadata
- -
- FAIL -
- - - - + PASS - + TestSTARTTLS 3 - 2 - 1 + 3 + 0 0 0 @@ -2454,33 +2289,11 @@ AssertionError: 30 != 0 - +
TestSTARTTLS - test_STARTTLS
- -
- FAIL -
- - - - + PASS @@ -4358,13 +4171,13 @@ AssertionError: 1 != 2 TestShelf + 17 16 - 15 0 0 1 - Detail + Detail @@ -4523,6 +4336,15 @@ AssertionError: 1 != 2 + +
TestShelf - test_shelf_order
+ + PASS + + + + +
TestShelf - test_xss_shelf
@@ -4558,13 +4380,13 @@ AssertionError: 1 != 2 TestSplitLibrary - 7 - 7 + 8 + 8 0 0 0 - Detail + Detail @@ -4632,6 +4454,15 @@ AssertionError: 1 != 2 + + + +
TestSplitLibrary - test_wrong_config_lib
+ + PASS + + + @@ -4883,14 +4714,14 @@ AssertionError: 1 != 2 - TestUploadEPubs - 6 - 6 + TestUploadAudio + 12 + 12 0 0 0 - Detail + Detail @@ -4898,7 +4729,7 @@ AssertionError: 1 != 2 -
TestUploadEPubs - test_upload_epub_comments
+
TestUploadAudio - test_upload_aac
PASS @@ -4907,7 +4738,7 @@ AssertionError: 1 != 2 -
TestUploadEPubs - test_upload_epub_cover
+
TestUploadAudio - test_upload_aiff
PASS @@ -4916,7 +4747,7 @@ AssertionError: 1 != 2 -
TestUploadEPubs - test_upload_epub_cover_formats
+
TestUploadAudio - test_upload_asf
PASS @@ -4925,7 +4756,7 @@ AssertionError: 1 != 2 -
TestUploadEPubs - test_upload_epub_duplicate
+
TestUploadAudio - test_upload_flac
PASS @@ -4934,7 +4765,7 @@ AssertionError: 1 != 2 -
TestUploadEPubs - test_upload_epub_identifier
+
TestUploadAudio - test_upload_m4a
PASS @@ -4942,6 +4773,129 @@ AssertionError: 1 != 2 + +
TestUploadAudio - test_upload_m4b
+ + PASS + + + + + + +
TestUploadAudio - test_upload_mp3
+ + PASS + + + + + + +
TestUploadAudio - test_upload_mp4
+ + PASS + + + + + + +
TestUploadAudio - test_upload_oggvorbis
+ + PASS + + + + + + +
TestUploadAudio - test_upload_ogv
+ + PASS + + + + + + +
TestUploadAudio - test_upload_opus
+ + PASS + + + + + + +
TestUploadAudio - test_upload_wav
+ + PASS + + + + + + + TestUploadEPubs + 6 + 6 + 0 + 0 + 0 + + Detail + + + + + + + +
TestUploadEPubs - test_upload_epub_comments
+ + PASS + + + + + + +
TestUploadEPubs - test_upload_epub_cover
+ + PASS + + + + + + +
TestUploadEPubs - test_upload_epub_cover_formats
+ + PASS + + + + + + +
TestUploadEPubs - test_upload_epub_duplicate
+ + PASS + + + + + + +
TestUploadEPubs - test_upload_epub_identifier
+ + PASS + + + + +
TestUploadEPubs - test_upload_epub_lang
@@ -4951,21 +4905,21 @@ AssertionError: 1 != 2 - + TestUserList 18 - 17 - 1 + 18 + 0 0 0 - Detail + Detail - +
TestUserList - test_edit_user_email
@@ -4974,7 +4928,7 @@ AssertionError: 1 != 2 - +
TestUserList - test_list_visibility
@@ -4983,7 +4937,7 @@ AssertionError: 1 != 2 - +
TestUserList - test_user_list_admin_role
@@ -4992,7 +4946,7 @@ AssertionError: 1 != 2 - +
TestUserList - test_user_list_check_sort
@@ -5001,7 +4955,7 @@ AssertionError: 1 != 2 - +
TestUserList - test_user_list_denied_tags
@@ -5010,7 +4964,7 @@ AssertionError: 1 != 2 - +
TestUserList - test_user_list_download_role
@@ -5019,36 +4973,16 @@ AssertionError: 1 != 2 - +
TestUserList - test_user_list_edit_button
- -
- FAIL -
- - - - + PASS - +
TestUserList - test_user_list_edit_email
@@ -5057,7 +4991,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_edit_kindle
@@ -5066,7 +5000,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_edit_language
@@ -5075,7 +5009,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_edit_locale
@@ -5084,7 +5018,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_edit_name
@@ -5093,7 +5027,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_edit_visiblility
@@ -5102,7 +5036,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_guest_edit
@@ -5111,7 +5045,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_remove_admin
@@ -5120,7 +5054,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_requests
@@ -5129,7 +5063,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_search
@@ -5138,7 +5072,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserList - test_user_list_sort
@@ -5156,13 +5090,13 @@ AssertionError: False is not true : Press cancel in User edit leads not back to 0 0 - Detail + Detail - +
TestUserLoad - test_user_change_vis
@@ -5180,13 +5114,13 @@ AssertionError: False is not true : Press cancel in User edit leads not back to 0 0 - Detail + Detail - +
TestUserTemplate - test_allow_column_restriction
@@ -5195,7 +5129,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_allow_tag_restriction
@@ -5204,7 +5138,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_archived_format_template
@@ -5213,7 +5147,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_author_user_template
@@ -5222,7 +5156,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_best_user_template
@@ -5231,7 +5165,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_category_user_template
@@ -5240,7 +5174,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_deny_column_restriction
@@ -5249,7 +5183,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_deny_tag_restriction
@@ -5258,7 +5192,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_detail_random_user_template
@@ -5267,7 +5201,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_download_user_template
@@ -5276,7 +5210,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_format_user_template
@@ -5285,7 +5219,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_hot_user_template
@@ -5294,7 +5228,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_language_user_template
@@ -5303,7 +5237,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_limit_book_languages
@@ -5312,7 +5246,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_list_user_template
@@ -5321,7 +5255,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_publisher_user_template
@@ -5330,7 +5264,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_random_user_template
@@ -5339,7 +5273,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_read_user_template
@@ -5348,7 +5282,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_recent_user_template
@@ -5357,7 +5291,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_series_user_template
@@ -5366,7 +5300,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestUserTemplate - test_ui_language_settings
@@ -5376,21 +5310,21 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - + TestCalibreWebVisibilitys 35 - 34 - 1 + 35 + 0 0 0 - Detail + Detail - +
TestCalibreWebVisibilitys - test_about
@@ -5399,7 +5333,7 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestCalibreWebVisibilitys - test_admin_SMTP_Settings
@@ -5408,36 +5342,16 @@ AssertionError: False is not true : Press cancel in User edit leads not back to - +
TestCalibreWebVisibilitys - test_admin_add_user
- -
- FAIL -
- - - - + PASS - +
TestCalibreWebVisibilitys - test_admin_change_password
@@ -5446,7 +5360,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_archived
@@ -5455,7 +5369,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_authors
@@ -5464,7 +5378,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_category
@@ -5473,7 +5387,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_file_formats
@@ -5482,7 +5396,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_hot
@@ -5491,7 +5405,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_language
@@ -5500,7 +5414,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_publisher
@@ -5509,7 +5423,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_random
@@ -5518,7 +5432,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_rated
@@ -5527,7 +5441,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_rating
@@ -5536,7 +5450,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_read
@@ -5545,7 +5459,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_admin_change_visibility_series
@@ -5554,7 +5468,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_allow_columns
@@ -5563,7 +5477,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_allow_tags
@@ -5572,7 +5486,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_archive_books
@@ -5581,7 +5495,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_authors_max_settings
@@ -5590,7 +5504,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_change_title
@@ -5599,7 +5513,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_checked_logged_in
@@ -5608,7 +5522,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_hide_custom_column
@@ -5617,7 +5531,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_link_column_to_read_status
@@ -5626,7 +5540,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_random_books_available
@@ -5635,7 +5549,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_read_status_visible
@@ -5644,7 +5558,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_request_link_column_to_read_status
@@ -5653,7 +5567,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_restrict_columns
@@ -5662,7 +5576,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_restrict_tags
@@ -5671,7 +5585,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_save_views_recent
@@ -5680,7 +5594,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_search_functions
@@ -5689,7 +5603,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_search_order
@@ -5698,7 +5612,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_search_string
@@ -5707,7 +5621,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_user_email_available
@@ -5716,7 +5630,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreWebVisibilitys - test_user_visibility_sidebar
@@ -5734,13 +5648,13 @@ AssertionError: 'Error creating new users' is not None 0 0 - Detail + Detail - +
TestCalibreHelper - test_author_sort
@@ -5749,7 +5663,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_author_sort_comma
@@ -5758,7 +5672,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_author_sort_junior
@@ -5767,7 +5681,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_author_sort_oneword
@@ -5776,7 +5690,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_author_sort_roman
@@ -5785,7 +5699,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_Limit_Length
@@ -5794,7 +5708,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_char_replacement
@@ -5803,7 +5717,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_chinese_Characters
@@ -5812,7 +5726,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_deg_eur_replacement
@@ -5821,7 +5735,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_doubleS
@@ -5830,7 +5744,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_finish_Dot
@@ -5839,7 +5753,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_high23
@@ -5848,7 +5762,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_check_umlauts
@@ -5857,7 +5771,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_random_password
@@ -5866,7 +5780,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_split_authors
@@ -5875,7 +5789,7 @@ AssertionError: 'Error creating new users' is not None - +
TestCalibreHelper - test_whitespaces
@@ -5886,11 +5800,11 @@ AssertionError: 'Error creating new users' is not None Total - 510 - 490 - 5 - 5 - 10 + 523 + 515 + 0 + 1 + 7   @@ -5918,7 +5832,7 @@ AssertionError: 'Error creating new users' is not None Platform - Linux 6.5.0-45-generic #45~22.04.1-Ubuntu SMP PREEMPT_DYNAMIC Mon Jul 15 16:40:02 UTC 2 x86_64 x86_64 + Linux 6.8.0-52-generic #53~22.04.1-Ubuntu SMP PREEMPT_DYNAMIC Wed Jan 15 19:18:46 UTC 2 x86_64 x86_64 Basic @@ -5928,21 +5842,15 @@ AssertionError: 'Error creating new users' is not None Basic - - advocate - 1.0.0 - Basic - - APScheduler - 3.10.4 + 3.11.0 Basic - Babel - 2.15.0 + babel + 2.17.0 Basic @@ -5954,13 +5862,19 @@ AssertionError: 'Error creating new users' is not None chardet - 4.0.0 + 5.2.0 + Basic + + + + cryptography + 44.0.2 Basic Flask - 3.0.3 + 3.1.0 Basic @@ -5978,7 +5892,7 @@ AssertionError: 'Error creating new users' is not None Flask-Limiter - 3.8.0 + 3.12 Basic @@ -5990,13 +5904,13 @@ AssertionError: 'Error creating new users' is not None Flask-WTF - 1.2.1 + 1.2.2 Basic greenlet - 3.0.3 + 3.1.1 Basic @@ -6008,25 +5922,25 @@ AssertionError: 'Error creating new users' is not None Jinja2 - 3.1.4 + 3.1.6 Basic lxml - 5.2.2 + 5.3.1 Basic - pyasn1 - 0.6.0 + netifaces-plus + 0.12.4 Basic pypdf - 4.2.0 + 5.4.0 Basic @@ -6038,31 +5952,31 @@ AssertionError: 'Error creating new users' is not None pytz - 2024.1 + 2025.2 Basic regex - 2024.5.15 + 2024.11.6 Basic requests - 2.31.0 + 2.32.3 Basic SQLAlchemy - 2.0.31 + 2.0.39 Basic tornado - 6.4.1 + 6.4.2 Basic @@ -6072,6 +5986,12 @@ AssertionError: 'Error creating new users' is not None Basic + + urllib3 + 2.3.0 + Basic + + Wand 0.6.13 @@ -6080,13 +6000,13 @@ AssertionError: 'Error creating new users' is not None Werkzeug - 3.0.3 + 3.1.3 Basic google-api-python-client - 2.139.0 + 2.165.0 TestBackupMetadataGdrive @@ -6104,19 +6024,19 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestBackupMetadataGdrive PyYAML - 6.0.1 + 6.0.2 TestBackupMetadataGdrive google-api-python-client - 2.139.0 + 2.165.0 TestCliGdrivedb @@ -6134,19 +6054,19 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestCliGdrivedb PyYAML - 6.0.1 + 6.0.2 TestCliGdrivedb google-api-python-client - 2.139.0 + 2.165.0 TestEbookConvertCalibreGDrive @@ -6164,19 +6084,19 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestEbookConvertCalibreGDrive PyYAML - 6.0.1 + 6.0.2 TestEbookConvertCalibreGDrive google-api-python-client - 2.139.0 + 2.165.0 TestEbookConvertGDriveKepubify @@ -6194,13 +6114,13 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestEbookConvertGDriveKepubify PyYAML - 6.0.1 + 6.0.2 TestEbookConvertGDriveKepubify @@ -6218,13 +6138,13 @@ AssertionError: 'Error creating new users' is not None py7zr - 0.21.1 + 0.22.0 TestEditBooks google-api-python-client - 2.139.0 + 2.165.0 TestEditAuthorsGdrive @@ -6242,25 +6162,25 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestEditAuthorsGdrive PyYAML - 6.0.1 + 6.0.2 TestEditAuthorsGdrive beautifulsoup4 - 4.12.3 + 4.13.3 TestLoadMetadata google-api-python-client - 2.139.0 + 2.165.0 TestEditBooksOnGdrive @@ -6278,19 +6198,19 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestEditBooksOnGdrive PyYAML - 6.0.1 + 6.0.2 TestEditBooksOnGdrive beautifulsoup4 - 4.12.3 + 4.13.3 TestLoadMetadataScholar @@ -6302,7 +6222,7 @@ AssertionError: 'Error creating new users' is not None google-api-python-client - 2.139.0 + 2.165.0 TestEmbedMetadataGdrive @@ -6320,19 +6240,19 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestEmbedMetadataGdrive PyYAML - 6.0.1 + 6.0.2 TestEmbedMetadataGdrive google-api-python-client - 2.139.0 + 2.165.0 TestSetupGdrive @@ -6350,13 +6270,13 @@ AssertionError: 'Error creating new users' is not None PyDrive2 - 1.20.0 + 1.21.3 TestSetupGdrive PyYAML - 6.0.1 + 6.0.2 TestSetupGdrive @@ -6368,7 +6288,7 @@ AssertionError: 'Error creating new users' is not None python-Levenshtein - 0.25.1 + 0.27.1 TestGoodreads @@ -6414,6 +6334,12 @@ AssertionError: 'Error creating new users' is not None TestOAuthLogin + + mutagen + 1.47.0 + TestUploadAudio + +
@@ -6422,7 +6348,7 @@ AssertionError: 'Error creating new users' is not None
diff --git a/test/Calibre-Web TestSummary_Windows.html b/test/Calibre-Web TestSummary_Windows.html deleted file mode 100644 index 21a41612..00000000 --- a/test/Calibre-Web TestSummary_Windows.html +++ /dev/null @@ -1,1027 +0,0 @@ - - - - Calibre-Web Tests - - - - - - - - - - - - - - - - - - - -
-

Calibre-Web Tests

-
-
-
-
-
-
- -
-
-
-
-
-
- -

Start Time: 2021-10-06 00:29:44

- -
-
-
-
- -

Stop Time: 2021-10-06 00:37:52

- -
-
-
-
-

Duration: 7:28 min

-
-
-
-
-
- -
-
-
-
-
- - - - - - -
-
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
Test Group/Test caseCountPassFailErrorSkipView
TestShelf2419131 - Detail -
-
TestShelf - test_add_shelf_from_search
-
PASS
-
TestShelf - test_adv_search_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_adv_search_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_arrange_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_arrange_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_delete_book_of_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_delete_book_of_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_private_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_private_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_public_private_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_public_private_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_public_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_public_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_rename_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_rename_shelf
-
-
- ERROR -
- - - -
-
TestShelf - test_shelf_action_non_shelf_edit_role
-
-
- FAIL -
- - - -
-
TestShelf - test_shelf_action_non_shelf_edit_role
-
-
- ERROR -
- - - -
-
TestShelf - test_shelf_anonymous
-
-
- ERROR -
- - - -
-
TestShelf - test_shelf_anonymous
-
-
- ERROR -
- - - -
-
TestShelf - test_shelf_database_change
-
-
- SKIP -
- - - -
-
TestShelf - test_shelf_long_name
-
-
- FAIL -
- - - -
-
TestShelf - test_shelf_long_name
-
-
- ERROR -
- - - -
-
TestShelf - test_xss_shelf
-
-
- FAIL -
- - - -
-
TestShelf - test_xss_shelf
-
-
- ERROR -
- - - -
Total2419131 
-
-
- -
-
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
Program libraryInstalled VersionTest class
PlatformWindows 10 10.0.19041 AMD64 Family 23 Model 113 Stepping 0, AuthenticAMD AMD64Basic
Python3.8.5Basic
APScheduler3.7.0Basic
Babel2.9.1Basic
backports-abc0.5Basic
Flask2.0.2Basic
Flask-Babel2.0.0Basic
Flask-Login0.5.0Basic
Flask-Principal0.4.0Basic
greenlet1.1.2Basic
iso-6390.4.5Basic
Jinja23.0.2Basic
lxml4.6.3Basic
PyPDF31.0.3Basic
pytz2021.3Basic
requests2.24.0Basic
six1.16.0Basic
SQLAlchemy1.4.25Basic
tornado6.1Basic
Unidecode1.2.0Basic
Wand0.6.7Basic
Werkzeug2.0.2Basic
-
-
-
-
- - - - - - - - - \ No newline at end of file