diff --git a/.github/workflows/cleanup_pr.yml b/.github/workflows/cleanup_pr.yml index 63090aba74a..20747030207 100644 --- a/.github/workflows/cleanup_pr.yml +++ b/.github/workflows/cleanup_pr.yml @@ -27,7 +27,7 @@ jobs: echo "##[set-output name=branch;]$(echo ${{ github.event.pull_request.head.ref }})" - name: Delete folder on builds.jabref.org if: ${{ steps.checksecrets.outputs.secretspresent }} - uses: appleboy/ssh-action@v0.1.4 + uses: appleboy/ssh-action@v0.1.5 with: script: rm -rf /var/www/builds.jabref.org/www/${{ steps.extract_branch.outputs.branch }} || true host: build-upload.jabref.org diff --git a/CHANGELOG.md b/CHANGELOG.md index 36ef36dedad..aa8854dcfe4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -16,6 +16,8 @@ Note that this project **does not** adhere to [Semantic Versioning](http://semve - We added an extra option in the 'Find Unlinked Files' dialog view to ignore unnecessary files like Thumbs.db, DS_Store, etc. [koppor#373](https://github.com/koppor/jabref/issues/373) - JabRef now writes log files. Linux: `$home/.cache/jabref/logs/version`, Windows: `%APPDATA%\..\Local\harawata\jabref\version\logs`, Mac: `Users/.../Library/Logs/jabref/version` - We added an importer for Citavi backup files, support ".ctv5bak" and ".ctv6bak" file formats. [#8322](https://github.com/JabRef/jabref/issues/8322) +- We added a feature to drag selected entries and drop them to other opened inactive library tabs [koppor521](https://github.com/koppor/jabref/issues/521). +- We added support for the [biblatex-apa](https://github.com/plk/biblatex-apa) legal entry types `Legislation`, `Legadminmaterial`, `Jurisdiction`, `Constitution` and `Legal` [#8931](https://github.com/JabRef/jabref/issues/8931) ### Changed @@ -35,7 +37,9 @@ Note that this project **does not** adhere to [Semantic Versioning](http://semve - When configured SSL certificates changed, JabRef warns the user to restart to apply the configuration. - We improved the appearances and logic of the "Manage field names & content" dialog, and renamed it to "Automatic field editor". [#6536](https://github.com/JabRef/jabref/issues/6536) - We improved the message explaining the options when modifying an automatic keyword group [#8911](https://github.com/JabRef/jabref/issues/8911) -- We moved the preferences option "Warn about duplicates on import" option from the tab "File" to the tab "Import and Export". [kopper#570](https://github.com/koppor/jabref/issues/570) +- We moved the preferences option "Warn about duplicates on import" option from the tab "File" to the tab "Import and Export". [koppor#570](https://github.com/koppor/jabref/issues/570) +- When JabRef encounters `% Encoding: UTF-8` header, it is kept during writing (and not removed). [#8964](https://github.com/JabRef/jabref/pull/8964) +- We replace characters which cannot be decoded using the specified encoding by a (probably another) valid character. This happens if JabRef detects the wrong charset (e.g., UTF-8 instead of Windows 1252). One can use the [Integrity Check](https://docs.jabref.org/finding-sorting-and-cleaning-entries/checkintegrity) to find those characters. ### Fixed @@ -54,6 +58,11 @@ Note that this project **does not** adhere to [Semantic Versioning](http://semve - We fixed an issue where removing several groups deletes only one of them. [#8390](https://github.com/JabRef/jabref/issues/8390) - We fixed an issue where the Sidepane (groups, web search and open office) width is not remembered after restarting JabRef. [#8907](https://github.com/JabRef/jabref/issues/8907) - We fixed a bug where switching between themes will cause an error/exception. [#8939](https://github.com/JabRef/jabref/pull/8939) +- We fixed a bug where files that were deleted in the source bibtex file were kept in the index. [#8962](https://github.com/JabRef/jabref/pull/8962) +- We fixed "Error while sending to JabRef" when the browser extension interacts with JabRef. [JabRef-Browser-Extension#479](https://github.com/JabRef/JabRef-Browser-Extension/issues/479) +- We fixed a bug where updating group view mode (intersection or union) requires re-selecting groups to take effect. [#6998](https://github.com/JabRef/jabref/issues/6998) +- We fixed a bug that prevented external group metadata changes from being merged. [#8873](https://github.com/JabRef/jabref/issues/8873) +- We fixed the shared database opening dialog to remember autosave folder and tick. [#7516](https://github.com/JabRef/jabref/issues/7516) ### Removed diff --git a/build.gradle b/build.gradle index 0cd6ae0c26d..0fc50f10ed7 100644 --- a/build.gradle +++ b/build.gradle @@ -85,7 +85,6 @@ repositories { } configurations { - antlr3 antlr4 // TODO: Remove the following workaround for split error messages such as // error: module java.xml.bind reads package javax.annotation from both jsr305 and java.annotation @@ -100,7 +99,7 @@ dependencyLocking { } javafx { - version = "18" + version = "18.0.1" modules = [ 'javafx.controls', 'javafx.fxml', 'javafx.web', 'javafx.swing' ] } @@ -116,33 +115,29 @@ dependencies { implementation 'org.apache.pdfbox:fontbox:3.0.0-RC1' implementation 'org.apache.pdfbox:xmpbox:3.0.0-RC1' - implementation 'org.apache.lucene:lucene-core:9.2.0' - implementation 'org.apache.lucene:lucene-queryparser:9.2.0' - implementation 'org.apache.lucene:lucene-queries:9.2.0' - implementation 'org.apache.lucene:lucene-analysis-common:9.2.0' - implementation 'org.apache.lucene:lucene-highlighter:9.2.0' + implementation 'org.apache.lucene:lucene-core:9.3.0' + implementation 'org.apache.lucene:lucene-queryparser:9.3.0' + implementation 'org.apache.lucene:lucene-queries:9.3.0' + implementation 'org.apache.lucene:lucene-analysis-common:9.3.0' + implementation 'org.apache.lucene:lucene-highlighter:9.3.0' implementation group: 'org.apache.commons', name: 'commons-csv', version: '1.9.0' implementation group: 'org.apache.commons', name: 'commons-lang3', version: '3.12.0' implementation 'com.h2database:h2-mvstore:2.1.214' implementation group: 'org.apache.tika', name: 'tika-core', version: '2.4.1' - implementation 'com.ibm.icu:icu4j-charset:71.1' // required for reading write-protected PDFs - see https://github.com/JabRef/jabref/pull/942#issuecomment-209252635 implementation 'org.bouncycastle:bcprov-jdk18on:1.71' implementation 'commons-cli:commons-cli:1.5.0' - implementation 'org.libreoffice:libreoffice:7.3.4' - implementation 'org.libreoffice:unoloader:7.3.4' + implementation 'org.libreoffice:libreoffice:7.3.5' + implementation 'org.libreoffice:unoloader:7.3.5' - implementation 'io.github.java-diff-utils:java-diff-utils:4.11' + implementation 'io.github.java-diff-utils:java-diff-utils:4.12' implementation 'info.debatty:java-string-similarity:2.0.0' - antlr3 'org.antlr:antlr:3.5.3' - implementation 'org.antlr:antlr-runtime:3.5.3' - antlr4 'org.antlr:antlr4:4.9.3' implementation 'org.antlr:antlr4-runtime:4.9.3' @@ -208,9 +203,9 @@ dependencies { implementation group: 'net.harawata', name: 'appdirs', version: '1.2.1' - testImplementation 'io.github.classgraph:classgraph:4.8.147' - testImplementation 'org.junit.jupiter:junit-jupiter:5.8.2' - testImplementation 'org.junit.platform:junit-platform-launcher:1.8.2' + testImplementation 'io.github.classgraph:classgraph:4.8.149' + testImplementation 'org.junit.jupiter:junit-jupiter:5.9.0' + testImplementation 'org.junit.platform:junit-platform-launcher:1.9.0' testImplementation 'org.mockito:mockito-core:4.6.1' testImplementation 'org.xmlunit:xmlunit-core:2.9.0' @@ -221,7 +216,7 @@ dependencies { testImplementation "org.testfx:testfx-junit5:4.0.16-alpha" testImplementation "org.hamcrest:hamcrest-library:2.2" - checkstyle 'com.puppycrawl.tools:checkstyle:10.3.1' + checkstyle 'com.puppycrawl.tools:checkstyle:10.3.2' // xjc needs the runtime as well for the ant task, otherwise it fails xjc group: 'org.glassfish.jaxb', name: 'jaxb-xjc', version: '3.0.2' xjc group: 'org.glassfish.jaxb', name: 'jaxb-runtime', version: '3.0.2' @@ -274,14 +269,14 @@ task generateSource(dependsOn: ["generateBstGrammarSource", } tasks.register("generateBstGrammarSource", JavaExec) { - main = "org.antlr.Tool" - classpath = configurations.antlr3 + main = "org.antlr.v4.Tool" + classpath = configurations.antlr4 group = "JabRef" - description = 'Generates BstLexer.java and BstParser.java from the Bst.g grammar file using antlr3.' + description = 'Generates BstLexer.java and BstParser.java from the Bst.g grammar file using antlr4.' - inputs.dir('src/main/antlr3/org/jabref/bst/') + inputs.dir('src/main/antlr4/org/jabref/bst/') outputs.dir("src-gen/main/java/org/jabref/logic/bst/") - args = ["-o", "src-gen/main/java/org/jabref/logic/bst/" , "$projectDir/src/main/antlr3/org/jabref/bst/Bst.g" ] + args = ["-o", "src-gen/main/java/org/jabref/logic/bst/", "-visitor", "-no-listener", "-package", "org.jabref.logic.bst", "$projectDir/src/main/antlr4/org/jabref/bst/Bst.g4"] } tasks.register("generateSearchGrammarSource", JavaExec) { diff --git a/buildres/csl/csl-locales/.github/workflows/merge.yaml b/buildres/csl/csl-locales/.github/workflows/merge.yaml index 0a0d9db955e..7e5736cc960 100644 --- a/buildres/csl/csl-locales/.github/workflows/merge.yaml +++ b/buildres/csl/csl-locales/.github/workflows/merge.yaml @@ -48,6 +48,10 @@ jobs: - deleted: [ '*.csl', 'dependent/*.csl', '*.xml' ] workflows: - added|modified: .github/workflows/*.yaml + updated_composer: + - added|modified: composer.json + deleted_composer: + - deleted: composer.json - name: Changed files if: github.event_name == 'push' @@ -87,11 +91,24 @@ jobs: run: cd release && git rm ${{ steps.update.outputs.deleted_files }} if: github.event_name == 'push' && steps.update.outputs.deleted == 'true' + - name: Update composer.json + if: steps.update.outputs.updated_composer == 'true' + run: | + cp composer.json release/composer.json + cd release + git add composer.json + + - name: Delete composer.json + if: steps.update.outputs.deleted_composer == 'true' + run: | + cd release + git rm composer.json + - uses: stefanzweifel/git-auto-commit-action@v4 with: repository: 'release' - commit_message: Releasing ${{ steps.update.outputs.updated_files }} ${{ steps.update.outputs.deleted_files }} - if: github.event_name == 'push' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true') + commit_message: Releasing ${{ steps.update.outputs.updated_files }} ${{ steps.update.outputs.deleted_files }} ${{ steps.update.outputs.updated_composer_files }} ${{ steps.update.outputs.deleted_composer_files }} + if: github.event_name == 'push' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true' || steps.update.outputs.updated_composer == 'true' || steps.update.outputs.deleted_composer == 'true') - uses: stefanzweifel/git-auto-commit-action@v4 with: @@ -116,3 +133,21 @@ jobs: git-user: "csl-bot" git-user-email: github@citationstyles.org git-commit-message: copied ${{ steps.update.outputs.workflows_files }} from styles + + - name: Bump version and push tag + id: tag_version + uses: mathieudutour/github-tag-action@v5.6 + with: + github_token: ${{ secrets.GITHUB_TOKEN }} + default_bump: patch + if: github.event_name == 'push' && hashFiles('composer.json') != '' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true') + + - name: Create a GitHub release + uses: softprops/action-gh-release@v0.1.14 + env: + github_token: ${{ secrets.GITHUB_TOKEN }} + with: + tag_name: ${{ steps.tag_version.outputs.new_tag }} + name: Release ${{ steps.tag_version.outputs.new_tag }} + body: Released ${{ steps.update.outputs.updated_files }} ${{ steps.update.outputs.deleted_files }} ${{ steps.update.outputs.updated_composer_files }} ${{ steps.update.outputs.deleted_composer_files }} + if: github.event_name == 'push' && hashFiles('composer.json') != '' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true') diff --git a/buildres/csl/csl-styles/.github/workflows/merge.yaml b/buildres/csl/csl-styles/.github/workflows/merge.yaml index 0a0d9db955e..7e5736cc960 100644 --- a/buildres/csl/csl-styles/.github/workflows/merge.yaml +++ b/buildres/csl/csl-styles/.github/workflows/merge.yaml @@ -48,6 +48,10 @@ jobs: - deleted: [ '*.csl', 'dependent/*.csl', '*.xml' ] workflows: - added|modified: .github/workflows/*.yaml + updated_composer: + - added|modified: composer.json + deleted_composer: + - deleted: composer.json - name: Changed files if: github.event_name == 'push' @@ -87,11 +91,24 @@ jobs: run: cd release && git rm ${{ steps.update.outputs.deleted_files }} if: github.event_name == 'push' && steps.update.outputs.deleted == 'true' + - name: Update composer.json + if: steps.update.outputs.updated_composer == 'true' + run: | + cp composer.json release/composer.json + cd release + git add composer.json + + - name: Delete composer.json + if: steps.update.outputs.deleted_composer == 'true' + run: | + cd release + git rm composer.json + - uses: stefanzweifel/git-auto-commit-action@v4 with: repository: 'release' - commit_message: Releasing ${{ steps.update.outputs.updated_files }} ${{ steps.update.outputs.deleted_files }} - if: github.event_name == 'push' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true') + commit_message: Releasing ${{ steps.update.outputs.updated_files }} ${{ steps.update.outputs.deleted_files }} ${{ steps.update.outputs.updated_composer_files }} ${{ steps.update.outputs.deleted_composer_files }} + if: github.event_name == 'push' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true' || steps.update.outputs.updated_composer == 'true' || steps.update.outputs.deleted_composer == 'true') - uses: stefanzweifel/git-auto-commit-action@v4 with: @@ -116,3 +133,21 @@ jobs: git-user: "csl-bot" git-user-email: github@citationstyles.org git-commit-message: copied ${{ steps.update.outputs.workflows_files }} from styles + + - name: Bump version and push tag + id: tag_version + uses: mathieudutour/github-tag-action@v5.6 + with: + github_token: ${{ secrets.GITHUB_TOKEN }} + default_bump: patch + if: github.event_name == 'push' && hashFiles('composer.json') != '' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true') + + - name: Create a GitHub release + uses: softprops/action-gh-release@v0.1.14 + env: + github_token: ${{ secrets.GITHUB_TOKEN }} + with: + tag_name: ${{ steps.tag_version.outputs.new_tag }} + name: Release ${{ steps.tag_version.outputs.new_tag }} + body: Released ${{ steps.update.outputs.updated_files }} ${{ steps.update.outputs.deleted_files }} ${{ steps.update.outputs.updated_composer_files }} ${{ steps.update.outputs.deleted_composer_files }} + if: github.event_name == 'push' && hashFiles('composer.json') != '' && (steps.update.outputs.updated == 'true' || steps.update.outputs.deleted == 'true') diff --git a/buildres/csl/csl-styles/annals-of-allergy-asthma-and-immunology.csl b/buildres/csl/csl-styles/annals-of-allergy-asthma-and-immunology.csl new file mode 100644 index 00000000000..b612c5ff55f --- /dev/null +++ b/buildres/csl/csl-styles/annals-of-allergy-asthma-and-immunology.csl @@ -0,0 +1,267 @@ + + diff --git a/buildres/csl/csl-styles/apa-5th-edition.csl b/buildres/csl/csl-styles/apa-5th-edition.csl index a2784ca4b5b..87cc83196a3 100644 --- a/buildres/csl/csl-styles/apa-5th-edition.csl +++ b/buildres/csl/csl-styles/apa-5th-edition.csl @@ -183,7 +183,7 @@ - + @@ -196,23 +196,37 @@ - + - + - + + + + + + + + + + + + diff --git a/buildres/csl/csl-styles/apa-6th-edition-no-ampersand.csl b/buildres/csl/csl-styles/apa-6th-edition-no-ampersand.csl index b0f618ec31f..ff55b46b98d 100644 --- a/buildres/csl/csl-styles/apa-6th-edition-no-ampersand.csl +++ b/buildres/csl/csl-styles/apa-6th-edition-no-ampersand.csl @@ -897,32 +897,52 @@ - + + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + - - + + + + + + + + + + + + + @@ -1401,7 +1421,7 @@ - diff --git a/buildres/csl/csl-styles/apa-6th-edition.csl b/buildres/csl/csl-styles/apa-6th-edition.csl index 775857accbe..851a3096ab0 100644 --- a/buildres/csl/csl-styles/apa-6th-edition.csl +++ b/buildres/csl/csl-styles/apa-6th-edition.csl @@ -896,32 +896,52 @@ - + + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + - - + + + + + + + + + + + + + @@ -1400,7 +1420,7 @@ - diff --git a/buildres/csl/csl-styles/apa-annotated-bibliography.csl b/buildres/csl/csl-styles/apa-annotated-bibliography.csl index 014ae44f162..0636d93db58 100644 --- a/buildres/csl/csl-styles/apa-annotated-bibliography.csl +++ b/buildres/csl/csl-styles/apa-annotated-bibliography.csl @@ -320,8 +320,8 @@ @@ -389,7 +389,7 @@ - @@ -470,7 +470,7 @@ - @@ -529,7 +529,7 @@ - @@ -1294,7 +1294,7 @@ - @@ -1559,7 +1559,7 @@ - + @@ -1567,6 +1567,20 @@ + + + + + + + + + + + @@ -1574,7 +1588,7 @@ - diff --git a/buildres/csl/csl-styles/apa-cv.csl b/buildres/csl/csl-styles/apa-cv.csl index 5a766c3dc61..228ad3ee882 100644 --- a/buildres/csl/csl-styles/apa-cv.csl +++ b/buildres/csl/csl-styles/apa-cv.csl @@ -320,8 +320,8 @@ @@ -415,7 +415,7 @@ - @@ -1085,7 +1085,7 @@ - @@ -1301,12 +1301,12 @@ + both publisher-place and event-place. Remove this 'choose' when that is changed. --> - + @@ -1314,6 +1314,20 @@ + + + + + + + + + + + @@ -1321,7 +1335,7 @@ - diff --git a/buildres/csl/csl-styles/apa-fr-provost.csl b/buildres/csl/csl-styles/apa-fr-provost.csl index 4b068cf752d..9e7928dca71 100644 --- a/buildres/csl/csl-styles/apa-fr-provost.csl +++ b/buildres/csl/csl-styles/apa-fr-provost.csl @@ -245,7 +245,7 @@ - + @@ -298,17 +298,17 @@ - + - + - + @@ -317,6 +317,20 @@ + + + + + + + + + + + diff --git a/buildres/csl/csl-styles/apa-no-ampersand.csl b/buildres/csl/csl-styles/apa-no-ampersand.csl index 446103abb69..5759e34ec4a 100644 --- a/buildres/csl/csl-styles/apa-no-ampersand.csl +++ b/buildres/csl/csl-styles/apa-no-ampersand.csl @@ -320,8 +320,8 @@ @@ -389,7 +389,7 @@ - @@ -470,7 +470,7 @@ - @@ -529,7 +529,7 @@ - @@ -1294,7 +1294,7 @@ - @@ -1554,12 +1554,12 @@ + both publisher-place and event-place. Remove this 'choose' when that is changed. --> - + @@ -1567,6 +1567,20 @@ + + + + + + + + + + + @@ -1574,7 +1588,7 @@ - diff --git a/buildres/csl/csl-styles/apa-no-doi-no-issue.csl b/buildres/csl/csl-styles/apa-no-doi-no-issue.csl index 08306362d76..c72c8bdf126 100644 --- a/buildres/csl/csl-styles/apa-no-doi-no-issue.csl +++ b/buildres/csl/csl-styles/apa-no-doi-no-issue.csl @@ -903,32 +903,52 @@ - + + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + - - + + + + + + + + + + + + + @@ -1414,7 +1434,7 @@ - diff --git a/buildres/csl/csl-styles/apa-no-initials.csl b/buildres/csl/csl-styles/apa-no-initials.csl index b8bcab11e79..4695cd9b6fb 100644 --- a/buildres/csl/csl-styles/apa-no-initials.csl +++ b/buildres/csl/csl-styles/apa-no-initials.csl @@ -320,8 +320,8 @@ @@ -389,7 +389,7 @@ - @@ -470,7 +470,7 @@ - @@ -529,7 +529,7 @@ - @@ -1294,7 +1294,7 @@ - @@ -1559,7 +1559,7 @@ - + @@ -1567,6 +1567,20 @@ + + + + + + + + + + + @@ -1574,7 +1588,7 @@ - diff --git a/buildres/csl/csl-styles/apa-numeric-superscript-brackets.csl b/buildres/csl/csl-styles/apa-numeric-superscript-brackets.csl index c6d3de36f82..ba3dadf6d20 100644 --- a/buildres/csl/csl-styles/apa-numeric-superscript-brackets.csl +++ b/buildres/csl/csl-styles/apa-numeric-superscript-brackets.csl @@ -123,8 +123,8 @@ @@ -192,7 +192,7 @@ - @@ -273,7 +273,7 @@ - @@ -332,7 +332,7 @@ - @@ -1097,7 +1097,7 @@ - @@ -1362,7 +1362,7 @@ - + @@ -1370,6 +1370,20 @@ + + + + + + + + + + + @@ -1377,7 +1391,7 @@ - diff --git a/buildres/csl/csl-styles/apa-numeric-superscript.csl b/buildres/csl/csl-styles/apa-numeric-superscript.csl index 0a6af38540b..1ed0ae88b7c 100644 --- a/buildres/csl/csl-styles/apa-numeric-superscript.csl +++ b/buildres/csl/csl-styles/apa-numeric-superscript.csl @@ -123,8 +123,8 @@ @@ -192,7 +192,7 @@ - @@ -273,7 +273,7 @@ - @@ -332,7 +332,7 @@ - @@ -1097,7 +1097,7 @@ - @@ -1362,7 +1362,7 @@ - + @@ -1370,6 +1370,20 @@ + + + + + + + + + + + @@ -1377,7 +1391,7 @@ - diff --git a/buildres/csl/csl-styles/apa-old-doi-prefix.csl b/buildres/csl/csl-styles/apa-old-doi-prefix.csl index 02ecfc75fac..cdfaa73a451 100644 --- a/buildres/csl/csl-styles/apa-old-doi-prefix.csl +++ b/buildres/csl/csl-styles/apa-old-doi-prefix.csl @@ -897,32 +897,52 @@ - + + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + - - + + + + + + + + + + + + + @@ -1401,7 +1421,7 @@ - diff --git a/buildres/csl/csl-styles/apa-single-spaced.csl b/buildres/csl/csl-styles/apa-single-spaced.csl index c2157cb296a..ff224f6a405 100644 --- a/buildres/csl/csl-styles/apa-single-spaced.csl +++ b/buildres/csl/csl-styles/apa-single-spaced.csl @@ -320,8 +320,8 @@ @@ -389,7 +389,7 @@ - @@ -470,7 +470,7 @@ - @@ -529,7 +529,7 @@ - @@ -1294,7 +1294,7 @@ - @@ -1554,12 +1554,12 @@ + both publisher-place and event-place. Remove this 'choose' when that is changed. --> - + @@ -1567,6 +1567,20 @@ + + + + + + + + + + + @@ -1574,7 +1588,7 @@ - diff --git a/buildres/csl/csl-styles/apa-tr.csl b/buildres/csl/csl-styles/apa-tr.csl index 557066dafd8..d17be770985 100644 --- a/buildres/csl/csl-styles/apa-tr.csl +++ b/buildres/csl/csl-styles/apa-tr.csl @@ -223,7 +223,7 @@ - + @@ -241,15 +241,15 @@ - + - + - + @@ -257,6 +257,20 @@ + + + + + + + + + + + diff --git a/buildres/csl/csl-styles/apa-with-abstract.csl b/buildres/csl/csl-styles/apa-with-abstract.csl index ddff715c9d6..25d185866bf 100644 --- a/buildres/csl/csl-styles/apa-with-abstract.csl +++ b/buildres/csl/csl-styles/apa-with-abstract.csl @@ -320,8 +320,8 @@ @@ -389,7 +389,7 @@ - @@ -470,7 +470,7 @@ - @@ -529,7 +529,7 @@ - @@ -1294,7 +1294,7 @@ - @@ -1559,7 +1559,7 @@ - + @@ -1567,6 +1567,20 @@ + + + + + + + + + + + @@ -1574,7 +1588,7 @@ - diff --git a/buildres/csl/csl-styles/apa.csl b/buildres/csl/csl-styles/apa.csl index 09f4635a505..0fd6bd71a5e 100644 --- a/buildres/csl/csl-styles/apa.csl +++ b/buildres/csl/csl-styles/apa.csl @@ -320,8 +320,8 @@ @@ -389,7 +389,7 @@ - @@ -470,7 +470,7 @@ - @@ -530,7 +530,7 @@ - @@ -1295,7 +1295,7 @@ - @@ -1560,7 +1560,7 @@ - + @@ -1568,6 +1568,20 @@ + + + + + + + + + + + @@ -1575,7 +1589,7 @@ - diff --git a/buildres/csl/csl-styles/associacao-brasileira-de-normas-tecnicas.csl b/buildres/csl/csl-styles/associacao-brasileira-de-normas-tecnicas.csl index 2b769703f27..cc8975fa8c2 100644 --- a/buildres/csl/csl-styles/associacao-brasileira-de-normas-tecnicas.csl +++ b/buildres/csl/csl-styles/associacao-brasileira-de-normas-tecnicas.csl @@ -82,7 +82,7 @@ tendo as inicias separadas por ponto.--> - + @@ -613,7 +613,7 @@ em caixa alta. Utiliza-se antes do nome da conferencia a expressao "In". Segundo - + diff --git a/buildres/csl/csl-styles/association-for-computational-linguistics.csl b/buildres/csl/csl-styles/association-for-computational-linguistics.csl index 63ea2e0f2a4..ec95b2a95b5 100644 --- a/buildres/csl/csl-styles/association-for-computational-linguistics.csl +++ b/buildres/csl/csl-styles/association-for-computational-linguistics.csl @@ -67,11 +67,6 @@ - - - - - @@ -199,7 +194,6 @@ - @@ -212,7 +206,6 @@ - @@ -236,7 +229,6 @@ - @@ -244,7 +236,6 @@ - @@ -262,18 +253,10 @@ - - - - - - - - - + @@ -282,7 +265,6 @@ - @@ -293,13 +275,11 @@ - - diff --git a/buildres/csl/csl-styles/australasian-journal-of-philosophy.csl b/buildres/csl/csl-styles/australasian-journal-of-philosophy.csl new file mode 100644 index 00000000000..ed846706972 --- /dev/null +++ b/buildres/csl/csl-styles/australasian-journal-of-philosophy.csl @@ -0,0 +1,317 @@ + + diff --git a/buildres/csl/csl-styles/beilstein-journal-of-organic-chemistry.csl b/buildres/csl/csl-styles/beilstein-journal-of-organic-chemistry.csl index 8487b04c1ee..2ab0fe83caf 100644 --- a/buildres/csl/csl-styles/beilstein-journal-of-organic-chemistry.csl +++ b/buildres/csl/csl-styles/beilstein-journal-of-organic-chemistry.csl @@ -3,7 +3,7 @@ Beilstein Journal of Organic Chemistry - Beilstein + BJOC http://www.zotero.org/styles/beilstein-journal-of-organic-chemistry @@ -154,7 +154,6 @@ - diff --git a/buildres/csl/csl-styles/bibliothek-forschung-und-praxis.csl b/buildres/csl/csl-styles/bibliothek-forschung-und-praxis.csl new file mode 100644 index 00000000000..5e85189c238 --- /dev/null +++ b/buildres/csl/csl-styles/bibliothek-forschung-und-praxis.csl @@ -0,0 +1,623 @@ + + diff --git a/buildres/csl/csl-styles/chaucer-review.csl b/buildres/csl/csl-styles/chaucer-review.csl new file mode 100644 index 00000000000..123910b954a --- /dev/null +++ b/buildres/csl/csl-styles/chaucer-review.csl @@ -0,0 +1,775 @@ + + diff --git a/buildres/csl/csl-styles/chemical-engineering-technology.csl b/buildres/csl/csl-styles/chemical-engineering-technology.csl new file mode 100644 index 00000000000..8497a7049e3 --- /dev/null +++ b/buildres/csl/csl-styles/chemical-engineering-technology.csl @@ -0,0 +1,173 @@ + + diff --git a/buildres/csl/csl-styles/composer.json b/buildres/csl/csl-styles/composer.json new file mode 100644 index 00000000000..7906f93ba75 --- /dev/null +++ b/buildres/csl/csl-styles/composer.json @@ -0,0 +1,13 @@ +{ + "name": "citation-style-language/styles", + "description": "Citation Style Language (CSL) Styless", + "type": "library", + "license": "CC-BY-SA-3.0", + "homepage": "http://citationstyles.org/", + "authors": [ + { + "name": "Citation Style Language (CSL) Team", + "homepage": "http://citationstyles.org/about/#Credits" + } + ] +} diff --git a/buildres/csl/csl-styles/copernicus-publications.csl b/buildres/csl/csl-styles/copernicus-publications.csl index 1745d371a3f..018ec7cb15c 100644 --- a/buildres/csl/csl-styles/copernicus-publications.csl +++ b/buildres/csl/csl-styles/copernicus-publications.csl @@ -255,7 +255,7 @@ - + diff --git a/buildres/csl/csl-styles/dependent/annals-of-allergy-asthma-and-immunology.csl b/buildres/csl/csl-styles/dependent/annals-of-allergy-asthma-and-immunology.csl deleted file mode 100644 index 12086db4e67..00000000000 --- a/buildres/csl/csl-styles/dependent/annals-of-allergy-asthma-and-immunology.csl +++ /dev/null @@ -1,14 +0,0 @@ - - diff --git a/buildres/csl/csl-styles/deutsche-gesellschaft-fur-psychologie.csl b/buildres/csl/csl-styles/deutsche-gesellschaft-fur-psychologie.csl index e6795a3b3e4..2f7e4e8243e 100644 --- a/buildres/csl/csl-styles/deutsche-gesellschaft-fur-psychologie.csl +++ b/buildres/csl/csl-styles/deutsche-gesellschaft-fur-psychologie.csl @@ -33,6 +33,7 @@ et al. Zugriff am Original erschienen + n. d. diff --git a/buildres/csl/csl-styles/entomological-review.csl b/buildres/csl/csl-styles/entomological-review.csl new file mode 100644 index 00000000000..7b078499259 --- /dev/null +++ b/buildres/csl/csl-styles/entomological-review.csl @@ -0,0 +1,231 @@ + + diff --git a/buildres/csl/csl-styles/haffner-style-manual.csl b/buildres/csl/csl-styles/haffner-style-manual.csl new file mode 100644 index 00000000000..279bbd8156a --- /dev/null +++ b/buildres/csl/csl-styles/haffner-style-manual.csl @@ -0,0 +1,512 @@ + + diff --git a/buildres/csl/csl-styles/health-physics.csl b/buildres/csl/csl-styles/health-physics.csl new file mode 100644 index 00000000000..dc69ba1cefe --- /dev/null +++ b/buildres/csl/csl-styles/health-physics.csl @@ -0,0 +1,161 @@ + + diff --git a/buildres/csl/csl-styles/journal-on-efficiency-and-responsibility-in-education-and-science.csl b/buildres/csl/csl-styles/journal-on-efficiency-and-responsibility-in-education-and-science.csl index 43ac15cd67d..e3387113c32 100644 --- a/buildres/csl/csl-styles/journal-on-efficiency-and-responsibility-in-education-and-science.csl +++ b/buildres/csl/csl-styles/journal-on-efficiency-and-responsibility-in-education-and-science.csl @@ -10,7 +10,7 @@ Dominik Bláha - blahad@sic.czu.cz + blahad@lib.czu.cz Igor Krejčí @@ -20,7 +20,7 @@ 2336-2375 1803-1617 - 2019-04-20T15:03:39+00:00 + 2022-07-10T17:58:51+00:00 This work is licensed under a Creative Commons Attribution-ShareAlike 3.0 License @@ -618,8 +618,8 @@ - - + + diff --git a/buildres/csl/csl-styles/nature-no-et-al.csl b/buildres/csl/csl-styles/nature-no-et-al.csl index 60dbfb6c761..115b680cb44 100644 --- a/buildres/csl/csl-styles/nature-no-et-al.csl +++ b/buildres/csl/csl-styles/nature-no-et-al.csl @@ -12,7 +12,7 @@ - 2014-06-22T12:00:00+00:00 + 2014-06-23T03:22:15+00:00 This work is licensed under a Creative Commons Attribution-ShareAlike 3.0 License @@ -35,7 +35,7 @@ - + @@ -55,6 +55,28 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/buildres/csl/csl-styles/nature-no-superscript.csl b/buildres/csl/csl-styles/nature-no-superscript.csl index 225eaf50c59..6c04d8139b3 100644 --- a/buildres/csl/csl-styles/nature-no-superscript.csl +++ b/buildres/csl/csl-styles/nature-no-superscript.csl @@ -12,7 +12,7 @@ - 2014-06-03T22:14:43+00:00 + 2022-07-02T11:44:17+00:00 This work is licensed under a Creative Commons Attribution-ShareAlike 3.0 License @@ -35,7 +35,7 @@ - + @@ -55,6 +55,28 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/buildres/csl/csl-styles/nature.csl b/buildres/csl/csl-styles/nature.csl index 2646cfe922c..a2253b8b817 100644 --- a/buildres/csl/csl-styles/nature.csl +++ b/buildres/csl/csl-styles/nature.csl @@ -15,7 +15,7 @@ 0028-0836 1476-4687 - 2019-10-08T13:18:12+00:00 + 2022-07-01T16:03:34+00:00 This work is licensed under a Creative Commons Attribution-ShareAlike 3.0 License @@ -37,7 +37,7 @@ - + @@ -55,6 +55,28 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/buildres/csl/csl-styles/renamed-styles.json b/buildres/csl/csl-styles/renamed-styles.json index d5396d84489..e30246291be 100644 --- a/buildres/csl/csl-styles/renamed-styles.json +++ b/buildres/csl/csl-styles/renamed-styles.json @@ -529,5 +529,6 @@ "investigaciones-de-historia-economica-economic-history-research": "investigaciones-de-historia-economica", "reach-reviews-in-human-space-exploration": "reach", "journal-of-oncological-science": "journal-of-oncological-sciences", - "harvard-dublin-city-university": "harvard-cite-them-right" + "harvard-dublin-city-university": "harvard-cite-them-right", + "technische-universitat-dresden-medizin": "apa" } diff --git a/buildres/csl/csl-styles/science.csl b/buildres/csl/csl-styles/science.csl index 641de8f592d..06ee7b70359 100644 --- a/buildres/csl/csl-styles/science.csl +++ b/buildres/csl/csl-styles/science.csl @@ -144,6 +144,7 @@ + diff --git a/buildres/csl/csl-styles/sciences-po-ecole-doctorale-author-date.csl b/buildres/csl/csl-styles/sciences-po-ecole-doctorale-author-date.csl index 8b570113983..3fe5189cd70 100644 --- a/buildres/csl/csl-styles/sciences-po-ecole-doctorale-author-date.csl +++ b/buildres/csl/csl-styles/sciences-po-ecole-doctorale-author-date.csl @@ -1,7 +1,7 @@ - diff --git a/buildres/csl/csl-styles/university-of-pretoria-harvard-theology-religion.csl b/buildres/csl/csl-styles/university-of-pretoria-harvard-theology-religion.csl new file mode 100644 index 00000000000..f59a9edfa0a --- /dev/null +++ b/buildres/csl/csl-styles/university-of-pretoria-harvard-theology-religion.csl @@ -0,0 +1,347 @@ + + diff --git a/buildres/csl/csl-styles/vancouver-author-date.csl b/buildres/csl/csl-styles/vancouver-author-date.csl index 13b1bc2d023..4bd51dd6b9d 100644 --- a/buildres/csl/csl-styles/vancouver-author-date.csl +++ b/buildres/csl/csl-styles/vancouver-author-date.csl @@ -7,6 +7,7 @@ + Charles Parnot charles.parnot@gmail.com @@ -15,7 +16,7 @@ Vancouver style as outlined by International Committee of Medical Journal Editors Uniform Requirements for Manuscripts Submitted to Biomedical Journals: Sample References - 2012-11-09T12:00:00+00:00 + 2022-07-18T16:49:43+00:00 This work is licensed under a Creative Commons Attribution-ShareAlike 3.0 License @@ -47,6 +48,16 @@ + + + + + + + + + + @@ -253,25 +264,29 @@ - - - - - + + + + + + - - - + + + + + + diff --git a/buildres/windows/JabRefHost.bat b/buildres/windows/JabRefHost.bat index 8561a8dedd1..1da03167e39 100644 --- a/buildres/windows/JabRefHost.bat +++ b/buildres/windows/JabRefHost.bat @@ -2,13 +2,14 @@ pushd %~dp0 :: Test if pwsh exists +setlocal enabledelayedexpansion where /q pwsh.exe -if %errorlevel%==0 ( +if !ERRORLEVEL!==0 ( @pwsh.exe -ExecutionPolicy Bypass -NoLogo -NonInteractive -NoProfile -WindowStyle Hidden -File ".\JabRefHost.ps1" ) else ( :: If not, test if powershell exists where /q powershell.exe - if %errorlevel%==0 ( + if !ERRORLEVEL!==0 ( @powershell.exe -ExecutionPolicy Bypass -NoLogo -NonInteractive -NoProfile -WindowStyle Hidden -File ".\JabRefHost.ps1" ) else ( echo "Could not find pwsh.exe or powershell.exe" 1>&2 @@ -16,3 +17,4 @@ if %errorlevel%==0 ( exit /b 1 ) ) +endlocal diff --git a/external-libraries.md b/external-libraries.md index a59bbaee1f9..e207b994cf5 100644 --- a/external-libraries.md +++ b/external-libraries.md @@ -398,6 +398,9 @@ Id: com.ibm.icu:* Project: International Components for Unicode URL: https://icu.unicode.org/ License: Unicode License (https://www.unicode.org/copyright.html) +Note: Our own fork https://github.com/JabRef/icu. Upstream PR: https://github.com/unicode-org/icu/pull/2127 +Path: lib/icu4j.jar +SourcePath: lib/ic4j-src.jar ``` ```yaml @@ -519,13 +522,14 @@ License: Apache-2.0 3. (on WSL) `sed 's/[^a-z]*//' < build/dependencies.txt | sed "s/\(.*\) .*/\1/" | grep -v "\->" | sort | uniq > build/dependencies-for-external-libraries.txt` ```text -com.fasterxml.jackson.core:jackson-annotations:2.13.2 -com.fasterxml.jackson.core:jackson-core:2.13.2 -com.fasterxml.jackson.core:jackson-databind:2.13.2 -com.fasterxml.jackson.dataformat:jackson-dataformat-yaml:2.13.2 -com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.13.2 -com.fasterxml.jackson:jackson-bom:2.13.2 -com.github.tomtung:latex2unicode_2.12:0.2.6 +com.fasterxml.jackson.core:jackson-annotations:2.13.3 +com.fasterxml.jackson.core:jackson-core:2.13.3 +com.fasterxml.jackson.core:jackson-databind:2.13.3 +com.fasterxml.jackson.dataformat:jackson-dataformat-yaml:2.13.3 +com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.13.3 +com.fasterxml.jackson:jackson-bom:2.13.3 +com.github.sialcasa.mvvmFX:mvvmfx-validation:f195849ca9 +com.github.tomtung:latex2unicode_2.13:0.3.2 com.google.code.gson:gson:2.9.0 com.google.errorprone:error_prone_annotations:2.11.0 com.google.guava:failureaccess:1.0.1 @@ -533,11 +537,9 @@ com.google.guava:guava:31.1-jre com.google.guava:listenablefuture:9999.0-empty-to-avoid-conflict-with-guava com.google.j2objc:j2objc-annotations:1.3 com.googlecode.javaewah:JavaEWAH:1.1.13 -com.h2database:h2-mvstore:2.1.212 -com.ibm.icu:icu4j-charset:70.1 -com.ibm.icu:icu4j:70.1 +com.h2database:h2-mvstore:2.1.214 com.jfoenix:jfoenix:9.0.10 -com.konghq:unirest-java:3.13.8 +com.konghq:unirest-java:3.13.10 com.microsoft.azure:applicationinsights-core:2.4.1 com.microsoft.azure:applicationinsights-logging-log4j2:2.4.1 com.oracle.ojdbc:ojdbc10:19.3.0.0 @@ -565,10 +567,9 @@ com.vladsch.flexmark:flexmark-util-visitor:0.64.0 com.vladsch.flexmark:flexmark-util:0.64.0 com.vladsch.flexmark:flexmark:0.64.0 commons-cli:commons-cli:1.5.0 -commons-codec:commons-codec:1.11 +commons-codec:commons-codec:1.15 commons-io:commons-io:2.11.0 commons-logging:commons-logging:1.2 -de.saxsys:mvvmfx-validation:1.9.0-SNAPSHOT de.saxsys:mvvmfx:1.8.0 de.undercouch:citeproc-java:3.0.0-alpha.6 eu.lestard:doc-annotations:0.2 @@ -581,7 +582,7 @@ net.java.dev.jna:jna-platform:5.6.0 net.java.dev.jna:jna:5.6.0 net.jcip:jcip-annotations:1.0 net.jodah:typetools:0.6.1 -org.antlr:antlr-runtime:3.5.2 +org.antlr:antlr-runtime:3.5.3 org.antlr:antlr4-runtime:4.9.3 org.apache.commons:commons-csv:1.9.0 org.apache.commons:commons-lang3:3.12.0 @@ -590,23 +591,22 @@ org.apache.httpcomponents:httpclient:4.5.13 org.apache.httpcomponents:httpcore-nio:4.4.13 org.apache.httpcomponents:httpcore:4.4.13 org.apache.httpcomponents:httpmime:4.5.13 -org.apache.lucene:lucene-analysis-common:9.1.0 -org.apache.lucene:lucene-backward-codecs:9.1.0 -org.apache.lucene:lucene-core:9.1.0 -org.apache.lucene:lucene-highlighter:9.1.0 -org.apache.lucene:lucene-queries:9.1.0 -org.apache.lucene:lucene-queryparser:9.1.0 -org.apache.lucene:lucene-sandbox:9.1.0 +org.apache.lucene:lucene-analysis-common:9.2.0 +org.apache.lucene:lucene-core:9.2.0 +org.apache.lucene:lucene-highlighter:9.2.0 +org.apache.lucene:lucene-queries:9.2.0 +org.apache.lucene:lucene-queryparser:9.2.0 +org.apache.lucene:lucene-sandbox:9.2.0 org.apache.pdfbox:fontbox:3.0.0-RC1 org.apache.pdfbox:pdfbox:3.0.0-RC1 org.apache.pdfbox:xmpbox:3.0.0-RC1 -org.apache.tika:tika-core:2.3.0 -org.bouncycastle:bcprov-jdk15on:1.70 +org.apache.tika:tika-core:2.4.1 +org.bouncycastle:bcprov-jdk18on:1.71 org.checkerframework:checker-qual:3.12.0 org.codehaus.mojo:animal-sniffer-annotations:1.18 org.controlsfx:controlsfx:11.1.1 -org.eclipse.jgit:org.eclipse.jgit:6.1.0.202203080745-r -org.fxmisc.flowless:flowless:0.6.9 +org.eclipse.jgit:org.eclipse.jgit:6.2.0.202206071550-r +org.fxmisc.flowless:flowless:0.6.10 org.fxmisc.richtext:richtextfx:0.10.9 org.fxmisc.undo:undofx:2.1.1 org.fxmisc.wellbehaved:wellbehavedfx:0.3.3 @@ -616,23 +616,23 @@ org.glassfish.jaxb:jaxb-runtime:3.0.2 org.glassfish.jaxb:txw2:3.0.2 org.jbibtex:jbibtex:1.0.19 org.jetbrains:annotations:15.0 -org.jsoup:jsoup:1.14.3 +org.jsoup:jsoup:1.15.1 org.kordamp.ikonli:ikonli-core:12.3.1 org.kordamp.ikonli:ikonli-javafx:12.3.1 org.kordamp.ikonli:ikonli-materialdesign2-pack:12.3.1 -org.libreoffice:libreoffice:7.3.2 -org.libreoffice:unoloader:7.3.2 -org.mariadb.jdbc:mariadb-java-client:2.7.5 -org.openjfx:javafx-base:18 -org.openjfx:javafx-controls:18 -org.openjfx:javafx-fxml:18 -org.openjfx:javafx-graphics:18 -org.openjfx:javafx-media:18 -org.openjfx:javafx-swing:18 -org.openjfx:javafx-web:18 -org.postgresql:postgresql:42.3.3 +org.libreoffice:libreoffice:7.3.4 +org.libreoffice:unoloader:7.3.4 +org.mariadb.jdbc:mariadb-java-client:2.7.6 +org.openjfx:javafx-base:18.0.1 +org.openjfx:javafx-controls:18.0.1 +org.openjfx:javafx-fxml:18.0.1 +org.openjfx:javafx-graphics:18.0.1 +org.openjfx:javafx-media:18.0.1 +org.openjfx:javafx-swing:18.0.1 +org.openjfx:javafx-web:18.0.1 +org.postgresql:postgresql:42.4.0 org.reactfx:reactfx:2.0-M5 -org.scala-lang:scala-library:2.12.8 +org.scala-lang:scala-library:2.13.8 org.slf4j:slf4j-api:2.0.0-alpha7 org.tinylog:slf4j-tinylog:2.4.1 org.tinylog:tinylog-api:2.4.1 diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar index 41d9927a4d4..249e5832f09 100644 Binary files a/gradle/wrapper/gradle-wrapper.jar and b/gradle/wrapper/gradle-wrapper.jar differ diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index 5c51a4acc9d..012d6d90445 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,6 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionSha256Sum=29e49b10984e585d8118b7d0bc452f944e386458df27371b49b4ac1dec4b7fda -distributionUrl=https\://services.gradle.org/distributions/gradle-7.4.2-bin.zip +distributionSha256Sum=cb87f222c5585bd46838ad4db78463a5c5f3d336e5e2b98dc7c0c586527351c2 +distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-bin.zip zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/gradlew b/gradlew index 1b6c787337f..a69d9cb6c20 100755 --- a/gradlew +++ b/gradlew @@ -205,6 +205,12 @@ set -- \ org.gradle.wrapper.GradleWrapperMain \ "$@" +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + # Use "xargs" to parse quoted args. # # With -n1 it outputs one arg per line, with the quotes and backslashes removed. diff --git a/gradlew.bat b/gradlew.bat index 107acd32c4e..f127cfd49d4 100644 --- a/gradlew.bat +++ b/gradlew.bat @@ -14,7 +14,7 @@ @rem limitations under the License. @rem -@if "%DEBUG%" == "" @echo off +@if "%DEBUG%"=="" @echo off @rem ########################################################################## @rem @rem Gradle startup script for Windows @@ -25,7 +25,7 @@ if "%OS%"=="Windows_NT" setlocal set DIRNAME=%~dp0 -if "%DIRNAME%" == "" set DIRNAME=. +if "%DIRNAME%"=="" set DIRNAME=. set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% @@ -40,7 +40,7 @@ if defined JAVA_HOME goto findJavaFromJavaHome set JAVA_EXE=java.exe %JAVA_EXE% -version >NUL 2>&1 -if "%ERRORLEVEL%" == "0" goto execute +if %ERRORLEVEL% equ 0 goto execute echo. echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. @@ -75,13 +75,15 @@ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar :end @rem End local scope for the variables with windows NT shell -if "%ERRORLEVEL%"=="0" goto mainEnd +if %ERRORLEVEL% equ 0 goto mainEnd :fail rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of rem the _cmd.exe /c_ return code! -if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 -exit /b 1 +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% :mainEnd if "%OS%"=="Windows_NT" endlocal diff --git a/lib/icu4j-src.jar b/lib/icu4j-src.jar new file mode 100644 index 00000000000..032936efdbd Binary files /dev/null and b/lib/icu4j-src.jar differ diff --git a/lib/icu4j.jar b/lib/icu4j.jar new file mode 100644 index 00000000000..207c4a6afc7 Binary files /dev/null and b/lib/icu4j.jar differ diff --git a/src/main/antlr3/org/jabref/bst/Bst.g b/src/main/antlr3/org/jabref/bst/Bst.g deleted file mode 100644 index 498621ad1a1..00000000000 --- a/src/main/antlr3/org/jabref/bst/Bst.g +++ /dev/null @@ -1,96 +0,0 @@ -grammar Bst; - -options { - output=AST; -} - -tokens { - IDLIST; - STACK; - ENTRY; - COMMANDS; -} - -// applies only to the parser: -@header {// Generated by ANTLR -package org.jabref.logic.bst;} - -// applies only to the lexer: -@lexer::header {// Generated by ANTLR -package org.jabref.logic.bst;} - -program : commands+ -> ^(COMMANDS commands+); - -commands - : STRINGS^ idList - | INTEGERS^ idList - | FUNCTION^ id stack - | MACRO^ id '{'! STRING '}'! - | READ^ - | EXECUTE^ '{'! function '}'! - | ITERATE^ '{'! function '}'! - | REVERSE^ '{'! function '}'! - | ENTRY^ idList0 idList0 idList0 - | SORT^; - -identifier - : IDENTIFIER; - -id - : '{'! identifier '}'!; - -idList - : '{' identifier+ '}' -> ^(IDLIST identifier+); - -idList0 - : '{' identifier* '}' -> ^(IDLIST identifier*); - -function - : '<' | '>' | '=' | '+' | '-' | ':=' | '*' | identifier; - -stack - : '{' stackitem+ '}' -> ^(STACK stackitem+); - -stackitem - : function - | STRING - | INTEGER - | QUOTED - | stack; - -STRINGS : 'STRINGS'; -INTEGERS : 'INTEGERS'; -FUNCTION : 'FUNCTION'; -EXECUTE : 'EXECUTE'; -SORT : 'SORT'; -ITERATE : 'ITERATE'; -REVERSE : 'REVERSE'; -ENTRY : 'ENTRY'; -READ : 'READ'; -MACRO : 'MACRO'; - -QUOTED - : '\'' IDENTIFIER; - -IDENTIFIER - : LETTER (LETTER|NUMERAL|'_')* ; - -fragment LETTER - : ('a'..'z'|'A'..'Z'|'.'|'$'); - -STRING - : '"' (~('"'))* '"'; - -INTEGER - : '#' ('+'|'-')? NUMERAL+ ; - -fragment NUMERAL - : ('0'..'9'); - -WS - : (' '|'\t'|'\n'|'\r')+ {_channel=99;} ; - -LINE_COMMENT - : '%' ~('\n'|'\r')* '\r'? '\n' {_channel=99;} - ; - diff --git a/src/main/antlr4/org/jabref/bst/Bst.g4 b/src/main/antlr4/org/jabref/bst/Bst.g4 new file mode 100644 index 00000000000..92b96ce18df --- /dev/null +++ b/src/main/antlr4/org/jabref/bst/Bst.g4 @@ -0,0 +1,85 @@ +grammar Bst; + +// Lexer + +STRINGS : 'STRINGS'; +INTEGERS : 'INTEGERS'; +FUNCTION : 'FUNCTION'; +EXECUTE : 'EXECUTE'; +SORT : 'SORT'; +ITERATE : 'ITERATE'; +REVERSE : 'REVERSE'; +ENTRY : 'ENTRY'; +READ : 'READ'; +MACRO : 'MACRO'; + +GT : '>'; +LT : '<'; +EQUAL : '='; +ASSIGN : ':='; +ADD : '+'; +SUB : '-'; +CONCAT : '*'; +LBRACE : '{'; +RBRACE : '}'; + +fragment LETTER : ('a'..'z'|'A'..'Z'|'.'|'$'); +fragment NUMERAL : ('0'..'9'); + +IDENTIFIER : LETTER (LETTER|NUMERAL|'_')*; +INTEGER : '#' ('+'|'-')? NUMERAL+; +QUOTED : '\'' IDENTIFIER; +STRING : '"' (~('"'))* '"'; + +WS: [ \r\n\t]+ -> skip; +LINE_COMMENT : '%' ~('\n'|'\r')* '\r'? '\n' -> skip; + +// Parser + +bstFile + : commands+ EOF + ; + +commands + : STRINGS ids=idListObl #stringsCommand + | INTEGERS ids=idListObl #integersCommand + | FUNCTION LBRACE id=identifier RBRACE function=stack #functionCommand + | MACRO LBRACE id=identifier RBRACE LBRACE repl=STRING RBRACE #macroCommand + | READ #readCommand + | EXECUTE LBRACE bstFunction RBRACE #executeCommand + | ITERATE LBRACE bstFunction RBRACE #iterateCommand + | REVERSE LBRACE bstFunction RBRACE #reverseCommand + | ENTRY idListOpt idListOpt idListOpt #entryCommand + | SORT #sortCommand + ; + +identifier + : IDENTIFIER + ; + +// Obligatory identifier list +idListObl + : LBRACE identifier+ RBRACE + ; + +// Optional identifier list +idListOpt + : LBRACE identifier* RBRACE + ; + +bstFunction + : LT | GT | EQUAL | ADD | SUB | ASSIGN | CONCAT + | identifier + ; + +stack + : LBRACE stackitem+ RBRACE + ; + +stackitem + : bstFunction + | STRING + | INTEGER + | QUOTED + | stack + ; diff --git a/src/main/java/module-info.java b/src/main/java/module-info.java index 751863df3e1..c05e10bda2b 100644 --- a/src/main/java/module-info.java +++ b/src/main/java/module-info.java @@ -85,7 +85,6 @@ requires org.mariadb.jdbc; uses org.mariadb.jdbc.credential.CredentialPlugin; requires org.apache.commons.lang3; - requires antlr.runtime; requires org.antlr.antlr4.runtime; requires org.fxmisc.flowless; requires org.apache.tika.core; diff --git a/src/main/java/org/jabref/gui/EntryTypeView.java b/src/main/java/org/jabref/gui/EntryTypeView.java index 10a95f3a5d0..165e1d7a380 100644 --- a/src/main/java/org/jabref/gui/EntryTypeView.java +++ b/src/main/java/org/jabref/gui/EntryTypeView.java @@ -28,6 +28,7 @@ import org.jabref.logic.l10n.Localization; import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.BibEntryType; +import org.jabref.model.entry.types.BiblatexAPAEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexSoftwareEntryTypeDefinitions; import org.jabref.model.entry.types.BibtexEntryTypeDefinitions; @@ -157,6 +158,7 @@ public void initialize() { .filter(e -> !recommendedEntries.contains(e)) .collect(Collectors.toList()); otherEntries.addAll(BiblatexSoftwareEntryTypeDefinitions.ALL); + otherEntries.addAll(BiblatexAPAEntryTypeDefinitions.ALL); } else { recommendedEntries = BibtexEntryTypeDefinitions.RECOMMENDED; otherEntries = BibtexEntryTypeDefinitions.ALL diff --git a/src/main/java/org/jabref/gui/JabRefFrame.java b/src/main/java/org/jabref/gui/JabRefFrame.java index 91d074e7e7c..834f7beb325 100644 --- a/src/main/java/org/jabref/gui/JabRefFrame.java +++ b/src/main/java/org/jabref/gui/JabRefFrame.java @@ -222,6 +222,54 @@ private void initDragAndDrop() { } else { tabbedPane.getTabs().remove(dndIndicator); } + // Accept drag entries from MainTable + if (event.getDragboard().hasContent(DragAndDropDataFormats.ENTRIES)) { + event.acceptTransferModes(TransferMode.COPY); + event.consume(); + } + }); + + this.getScene().setOnDragEntered(event -> { + // It is necessary to setOnDragOver for newly opened tabs + // drag'n'drop on tabs covered dnd on tabbedPane, so dnd on tabs should contain all dnds on tabbedPane + tabbedPane.lookupAll(".tab").forEach(tab -> { + tab.setOnDragOver(tabDragEvent -> { + if (DragAndDropHelper.hasBibFiles(tabDragEvent.getDragboard())) { + tabDragEvent.acceptTransferModes(TransferMode.ANY); + if (!tabbedPane.getTabs().contains(dndIndicator)) { + tabbedPane.getTabs().add(dndIndicator); + } + event.consume(); + } else { + tabbedPane.getTabs().remove(dndIndicator); + } + + if (tabDragEvent.getDragboard().hasContent(DragAndDropDataFormats.ENTRIES)) { + tabDragEvent.acceptTransferModes(TransferMode.COPY); + tabDragEvent.consume(); + } + }); + tab.setOnDragExited(event1 -> tabbedPane.getTabs().remove(dndIndicator)); + tab.setOnDragDropped(tabDragEvent -> { + if (DragAndDropHelper.hasBibFiles(tabDragEvent.getDragboard())) { + tabbedPane.getTabs().remove(dndIndicator); + List bibFiles = DragAndDropHelper.getBibFiles(tabDragEvent.getDragboard()); + OpenDatabaseAction openDatabaseAction = this.getOpenDatabaseAction(); + openDatabaseAction.openFiles(bibFiles, true); + tabDragEvent.setDropCompleted(true); + tabDragEvent.consume(); + } else { + for (Tab libraryTab : tabbedPane.getTabs()) { + if (libraryTab.getId().equals(tab.getId()) && + !tabbedPane.getSelectionModel().getSelectedItem().equals(libraryTab)) { + ((LibraryTab) libraryTab).dropEntry(stateManager.getLocalDragboard().getBibEntries()); + } + } + tabDragEvent.consume(); + } + }); + }); + event.consume(); }); this.getScene().setOnDragExited(event -> tabbedPane.getTabs().remove(dndIndicator)); diff --git a/src/main/java/org/jabref/gui/LibraryTab.java b/src/main/java/org/jabref/gui/LibraryTab.java index c62741b37c1..139cdf31fd0 100644 --- a/src/main/java/org/jabref/gui/LibraryTab.java +++ b/src/main/java/org/jabref/gui/LibraryTab.java @@ -7,6 +7,7 @@ import java.util.List; import java.util.Objects; import java.util.Optional; +import java.util.Random; import javafx.animation.PauseTransition; import javafx.application.Platform; @@ -153,6 +154,10 @@ public LibraryTab(JabRefFrame frame, this.entryEditor = new EntryEditor(this, externalFileTypes); + // set LibraryTab ID for drag'n'drop + // ID content doesn't matter, we only need different tabs to have different ID + this.setId(Long.valueOf(new Random().nextLong()).toString()); + Platform.runLater(() -> { EasyBind.subscribe(changedProperty, this::updateTabTitle); stateManager.getOpenDatabases().addListener((ListChangeListener) c -> @@ -758,6 +763,10 @@ public void paste() { mainTable.paste(); } + public void dropEntry(List entriesToAdd) { + mainTable.dropEntry(entriesToAdd); + } + public void cut() { mainTable.cut(); } @@ -857,7 +866,7 @@ private class IndexUpdateListener { public IndexUpdateListener() { if (preferencesService.getFilePreferences().shouldFulltextIndexLinkedFiles()) { try { - indexingTaskManager.addToIndex(PdfIndexer.of(bibDatabaseContext, preferencesService.getFilePreferences()), bibDatabaseContext); + indexingTaskManager.updateIndex(PdfIndexer.of(bibDatabaseContext, preferencesService.getFilePreferences()), bibDatabaseContext); } catch (IOException e) { LOGGER.error("Cannot access lucene index", e); } diff --git a/src/main/java/org/jabref/gui/StateManager.java b/src/main/java/org/jabref/gui/StateManager.java index a8d93152de9..cc6ab8ae8bd 100644 --- a/src/main/java/org/jabref/gui/StateManager.java +++ b/src/main/java/org/jabref/gui/StateManager.java @@ -8,9 +8,11 @@ import javafx.beans.Observable; import javafx.beans.binding.Bindings; import javafx.beans.property.IntegerProperty; +import javafx.beans.property.ObjectProperty; import javafx.beans.property.ReadOnlyListProperty; import javafx.beans.property.ReadOnlyListWrapper; import javafx.beans.property.SimpleIntegerProperty; +import javafx.beans.property.SimpleObjectProperty; import javafx.collections.FXCollections; import javafx.collections.ObservableList; import javafx.collections.ObservableMap; @@ -18,6 +20,7 @@ import javafx.scene.Node; import javafx.util.Pair; +import org.jabref.gui.edit.automaticfiededitor.LastAutomaticFieldEditorEdit; import org.jabref.gui.sidepane.SidePaneType; import org.jabref.gui.util.BackgroundTask; import org.jabref.gui.util.CustomLocalDragboard; @@ -62,6 +65,8 @@ public class StateManager { private final ObservableMap dialogWindowStates = FXCollections.observableHashMap(); private final ObservableList visibleSidePanes = FXCollections.observableArrayList(); + private final ObjectProperty lastAutomaticFieldEditorEdit = new SimpleObjectProperty<>(); + public StateManager() { activeGroups.bind(Bindings.valueAt(selectedGroups, activeDatabase.orElse(null))); } @@ -172,4 +177,16 @@ public DialogWindowState getDialogWindowState(String className) { public void setDialogWindowState(String className, DialogWindowState state) { dialogWindowStates.put(className, state); } + + public ObjectProperty lastAutomaticFieldEditorEditProperty() { + return lastAutomaticFieldEditorEdit; + } + + public LastAutomaticFieldEditorEdit getLastAutomaticFieldEditorEdit() { + return lastAutomaticFieldEditorEditProperty().get(); + } + + public void setLastAutomaticFieldEditorEdit(LastAutomaticFieldEditorEdit automaticFieldEditorEdit) { + lastAutomaticFieldEditorEditProperty().set(automaticFieldEditorEdit); + } } diff --git a/src/main/java/org/jabref/gui/collab/ChangeDisplayDialog.java b/src/main/java/org/jabref/gui/collab/ChangeDisplayDialog.java deleted file mode 100644 index e9f0751cc07..00000000000 --- a/src/main/java/org/jabref/gui/collab/ChangeDisplayDialog.java +++ /dev/null @@ -1,114 +0,0 @@ -package org.jabref.gui.collab; - -import java.util.List; - -import javafx.collections.FXCollections; -import javafx.scene.control.Button; -import javafx.scene.control.ButtonBar; -import javafx.scene.control.ButtonBar.ButtonData; -import javafx.scene.control.ButtonType; -import javafx.scene.control.CheckBox; -import javafx.scene.control.Label; -import javafx.scene.control.ListView; -import javafx.scene.control.ScrollPane; -import javafx.scene.control.SplitPane; -import javafx.scene.layout.BorderPane; -import javafx.scene.layout.Region; -import javafx.scene.layout.VBox; - -import org.jabref.gui.undo.NamedCompound; -import org.jabref.gui.util.BaseDialog; -import org.jabref.logic.l10n.Localization; -import org.jabref.model.database.BibDatabaseContext; - -import com.tobiasdiez.easybind.EasyBind; - -class ChangeDisplayDialog extends BaseDialog { - - private final ListView changesList; - private final BorderPane infoPanel = new BorderPane(); - private final CheckBox cb = new CheckBox(Localization.lang("Accept change")); - - public ChangeDisplayDialog(BibDatabaseContext database, List changes) { - this.setTitle(Localization.lang("External changes")); - this.getDialogPane().setPrefSize(800, 600); - - changesList = new ListView<>(FXCollections.observableArrayList(changes)); - changesList.setPrefWidth(200); - EasyBind.subscribe(changesList.getSelectionModel().selectedItemProperty(), this::selectedChangeChanged); - - SplitPane pane = new SplitPane(); - pane.setDividerPositions(0.2); - - Button selectAllChangesFromDisk = new Button(Localization.lang("Mark all changes as accepted")); - selectAllChangesFromDisk.setMinWidth(Region.USE_PREF_SIZE); - selectAllChangesFromDisk.setOnAction(evt -> { - for (DatabaseChangeViewModel change : changes) { - change.setAccepted(true); - } - }); - Button unselectAllAcceptChanges = new Button(Localization.lang("Unmark all changes")); - unselectAllAcceptChanges.setOnAction(evt -> { - for (DatabaseChangeViewModel change : changes) { - change.setAccepted(false); - } - }); - - VBox leftContent = new VBox(changesList, - selectAllChangesFromDisk, - unselectAllAcceptChanges); - - ScrollPane leftScroll = new ScrollPane(leftContent); - leftScroll.setFitToHeight(true); - leftScroll.setFitToWidth(true); - - pane.getItems().addAll(leftScroll, infoPanel); - SplitPane.setResizableWithParent(leftScroll, false); - - getDialogPane().setContent(pane); - - Label rootInfo = new Label(Localization.lang("Select the tree nodes to view and accept or reject changes") + '.'); - infoPanel.setCenter(rootInfo); - changesList.getSelectionModel().selectFirst(); - - ButtonType dismissChanges = new ButtonType(Localization.lang("Dismiss"), ButtonData.CANCEL_CLOSE); - - getDialogPane().getButtonTypes().setAll(new ButtonType(Localization.lang("Accept changes"), ButtonBar.ButtonData.APPLY), - dismissChanges); - - setResultConverter(button -> { - if (button == dismissChanges) { - return false; - } else { - // Perform all accepted changes - NamedCompound ce = new NamedCompound(Localization.lang("Merged external changes")); - for (DatabaseChangeViewModel change : changes) { - if (change instanceof EntryChangeViewModel) { - // We don't have a checkbox for accept and always get the correct merged entry, the accept property in this special case only controls the radio buttons selection - change.makeChange(database, ce); - } else if (change.isAccepted()) { - change.makeChange(database, ce); - } - } - ce.end(); - // TODO: panel.getUndoManager().addEdit(ce); - - return true; - } - }); - } - - private void selectedChangeChanged(DatabaseChangeViewModel currentChange) { - if (currentChange != null) { - infoPanel.setCenter(currentChange.description()); - - if (!(currentChange instanceof EntryChangeViewModel)) { - cb.setManaged(true); - infoPanel.setBottom(cb); - cb.selectedProperty().bindBidirectional(currentChange.acceptedProperty()); - } else { - cb.setManaged(false); - } - } - } -} diff --git a/src/main/java/org/jabref/gui/collab/ChangeScanner.java b/src/main/java/org/jabref/gui/collab/ChangeScanner.java index f19f406e18f..61e44c9116e 100644 --- a/src/main/java/org/jabref/gui/collab/ChangeScanner.java +++ b/src/main/java/org/jabref/gui/collab/ChangeScanner.java @@ -99,6 +99,6 @@ private DatabaseChangeViewModel createBibEntryDiff(BibEntryDiff diff) { return new EntryDeleteChangeViewModel(diff.getOriginalEntry()); } - return new EntryChangeViewModel(diff.getOriginalEntry(), diff.getNewEntry()); + return new EntryChangeViewModel(diff.getOriginalEntry(), diff.getNewEntry(), dialogService); } } diff --git a/src/main/java/org/jabref/gui/collab/DatabaseChangeMonitor.java b/src/main/java/org/jabref/gui/collab/DatabaseChangeMonitor.java index 575307bcda8..20263b36ab6 100644 --- a/src/main/java/org/jabref/gui/collab/DatabaseChangeMonitor.java +++ b/src/main/java/org/jabref/gui/collab/DatabaseChangeMonitor.java @@ -67,7 +67,7 @@ public DatabaseChangeMonitor(BibDatabaseContext database, Localization.lang("The library has been modified by another program."), List.of(new Action(Localization.lang("Dismiss changes"), event -> notificationPane.hide()), new Action(Localization.lang("Review changes"), event -> { - dialogService.showCustomDialogAndWait(new ChangeDisplayDialog(database, changes)); + dialogService.showCustomDialogAndWait(new ExternalChangesResolverDialog(database, changes)); notificationPane.hide(); })), Duration.ZERO)); diff --git a/src/main/java/org/jabref/gui/collab/DatabaseChangeViewModel.java b/src/main/java/org/jabref/gui/collab/DatabaseChangeViewModel.java index bdc7dead217..d1302c39cc7 100644 --- a/src/main/java/org/jabref/gui/collab/DatabaseChangeViewModel.java +++ b/src/main/java/org/jabref/gui/collab/DatabaseChangeViewModel.java @@ -1,28 +1,29 @@ package org.jabref.gui.collab; +import java.util.Optional; + import javafx.beans.property.BooleanProperty; +import javafx.beans.property.ReadOnlyStringProperty; import javafx.beans.property.SimpleBooleanProperty; +import javafx.beans.property.SimpleStringProperty; +import javafx.beans.property.StringProperty; import javafx.scene.Node; +import org.jabref.gui.actions.SimpleCommand; import org.jabref.gui.undo.NamedCompound; import org.jabref.model.database.BibDatabaseContext; abstract class DatabaseChangeViewModel { - - protected String name; + private final StringProperty name = new SimpleStringProperty(""); private BooleanProperty acceptedProperty = new SimpleBooleanProperty(true); - DatabaseChangeViewModel() { - name = ""; - } - DatabaseChangeViewModel(String name) { - this.name = name; + setName(name); } @Override public String toString() { - return name; + return getName(); } public boolean isAccepted() { @@ -37,6 +38,18 @@ public void setAccepted(boolean accepted) { this.acceptedProperty.setValue(accepted); } + public ReadOnlyStringProperty nameProperty() { + return name; + } + + public String getName() { + return nameProperty().get(); + } + + private void setName(String str) { + name.set(str); + } + /** * This method returns a {@link Node} detailing the nature and look of the change, e.g. how it is displayed * @@ -51,4 +64,12 @@ public void setAccepted(boolean accepted) { * @param undoEdit NamedCompound The compound to hold the undo edits. */ public abstract void makeChange(BibDatabaseContext database, NamedCompound undoEdit); + + public boolean hasAdvancedMergeDialog() { + return false; + } + + public Optional openAdvancedMergeDialog() { + return Optional.empty(); + } } diff --git a/src/main/java/org/jabref/gui/collab/EntryAddChangeViewModel.java b/src/main/java/org/jabref/gui/collab/EntryAddChangeViewModel.java index 56d763be3c6..3f7be35d198 100644 --- a/src/main/java/org/jabref/gui/collab/EntryAddChangeViewModel.java +++ b/src/main/java/org/jabref/gui/collab/EntryAddChangeViewModel.java @@ -26,14 +26,13 @@ public EntryAddChangeViewModel(BibEntry entry, DialogService dialogService, StateManager stateManager, ThemeManager themeManager) { - super(); + super(entry.getCitationKey() + .map(key -> Localization.lang("Added entry") + ": '" + key + '\'') + .orElse(Localization.lang("Added entry"))); this.dialogService = dialogService; this.preferencesService = preferencesService; this.themeManager = themeManager; this.stateManager = stateManager; - this.name = entry.getCitationKey() - .map(key -> Localization.lang("Added entry") + ": '" + key + '\'') - .orElse(Localization.lang("Added entry")); this.entry = entry; } diff --git a/src/main/java/org/jabref/gui/collab/EntryChangeViewModel.java b/src/main/java/org/jabref/gui/collab/EntryChangeViewModel.java index 421e4c1fcd9..042a4636a61 100644 --- a/src/main/java/org/jabref/gui/collab/EntryChangeViewModel.java +++ b/src/main/java/org/jabref/gui/collab/EntryChangeViewModel.java @@ -1,10 +1,16 @@ package org.jabref.gui.collab; +import java.util.Optional; + import javafx.geometry.Insets; import javafx.scene.Node; import javafx.scene.control.Label; import javafx.scene.layout.VBox; +import org.jabref.gui.DialogService; +import org.jabref.gui.actions.SimpleCommand; +import org.jabref.gui.mergeentries.MergeEntriesDialog; +import org.jabref.gui.mergeentries.MergeTwoEntriesAction; import org.jabref.gui.mergeentries.newmergedialog.ShowDiffConfig; import org.jabref.gui.mergeentries.newmergedialog.ThreeWayMergeView; import org.jabref.gui.mergeentries.newmergedialog.diffhighlighter.DiffHighlighter; @@ -21,15 +27,15 @@ class EntryChangeViewModel extends DatabaseChangeViewModel { private final BibEntry newEntry; private ThreeWayMergeView threeWayMergeView; - public EntryChangeViewModel(BibEntry entry, BibEntry newEntry) { - super(); + private final DialogService dialogService; + + public EntryChangeViewModel(BibEntry entry, BibEntry newEntry, DialogService dialogService) { + super(entry.getCitationKey().map(key -> Localization.lang("Modified entry") + ": '" + key + '\'') + .orElse(Localization.lang("Modified entry"))); this.oldEntry = entry; this.newEntry = newEntry; - - name = entry.getCitationKey() - .map(key -> Localization.lang("Modified entry") + ": '" + key + '\'') - .orElse(Localization.lang("Modified entry")); + this.dialogService = dialogService; } /** @@ -47,13 +53,10 @@ public void setAccepted(boolean accepted) { @Override public void makeChange(BibDatabaseContext database, NamedCompound undoEdit) { - this.description(); // Init dialog to prevent NPE database.getDatabase().removeEntry(oldEntry); - BibEntry mergedEntry = threeWayMergeView.getMergedEntry(); - mergedEntry.setId(oldEntry.getId()); // Keep ID - database.getDatabase().insertEntry(mergedEntry); + database.getDatabase().insertEntry(newEntry); undoEdit.addEdit(new UndoableInsertEntries(database.getDatabase(), oldEntry)); - undoEdit.addEdit(new UndoableInsertEntries(database.getDatabase(), mergedEntry)); + undoEdit.addEdit(new UndoableInsertEntries(database.getDatabase(), newEntry)); } @Override @@ -62,11 +65,23 @@ public Node description() { threeWayMergeView.selectLeftEntryValues(); threeWayMergeView.showDiff(new ShowDiffConfig(ThreeWayMergeToolbar.DiffView.SPLIT, DiffHighlighter.DiffMethod.WORDS)); VBox container = new VBox(10); - Label header = new Label(name); + Label header = new Label(getName()); header.getStyleClass().add("sectionHeader"); container.getChildren().add(header); container.getChildren().add(threeWayMergeView); VBox.setMargin(threeWayMergeView, new Insets(5, 5, 5, 5)); return container; } + + @Override + public boolean hasAdvancedMergeDialog() { + return true; + } + + @Override + public Optional openAdvancedMergeDialog() { + MergeEntriesDialog mergeEntriesDialog = new MergeEntriesDialog(oldEntry, newEntry); + return dialogService.showCustomDialogAndWait(mergeEntriesDialog) + .map(res -> new MergeTwoEntriesAction(res, null, null)); + } } diff --git a/src/main/java/org/jabref/gui/collab/EntryDeleteChangeViewModel.java b/src/main/java/org/jabref/gui/collab/EntryDeleteChangeViewModel.java index f30b6bda14a..3ccf47f8fc8 100644 --- a/src/main/java/org/jabref/gui/collab/EntryDeleteChangeViewModel.java +++ b/src/main/java/org/jabref/gui/collab/EntryDeleteChangeViewModel.java @@ -20,11 +20,9 @@ class EntryDeleteChangeViewModel extends DatabaseChangeViewModel { private final BibEntry entry; public EntryDeleteChangeViewModel(BibEntry entry) { - super(Localization.lang("Deleted entry")); - - this.name = entry.getCitationKey() - .map(key -> Localization.lang("Deleted entry") + ": '" + key + '\'') - .orElse(Localization.lang("Deleted entry")); + super(entry.getCitationKey() + .map(key -> Localization.lang("Deleted entry") + ": '" + key + '\'') + .orElse(Localization.lang("Deleted entry"))); this.entry = entry; } @@ -37,6 +35,7 @@ public void makeChange(BibDatabaseContext database, NamedCompound undoEdit) { @Override public Node description() { PreviewViewer previewViewer = new PreviewViewer(new BibDatabaseContext(), JabRefGUI.getMainFrame().getDialogService(), Globals.stateManager, Globals.getThemeManager()); + previewViewer.setLayout(Globals.prefs.getPreviewPreferences().getSelectedPreviewLayout()); previewViewer.setEntry(entry); return previewViewer; } diff --git a/src/main/java/org/jabref/gui/collab/ExternalChangesResolverDialog.fxml b/src/main/java/org/jabref/gui/collab/ExternalChangesResolverDialog.fxml new file mode 100644 index 00000000000..20e9f334412 --- /dev/null +++ b/src/main/java/org/jabref/gui/collab/ExternalChangesResolverDialog.fxml @@ -0,0 +1,46 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/main/java/org/jabref/gui/collab/ExternalChangesResolverDialog.java b/src/main/java/org/jabref/gui/collab/ExternalChangesResolverDialog.java new file mode 100644 index 00000000000..d3f69db2a05 --- /dev/null +++ b/src/main/java/org/jabref/gui/collab/ExternalChangesResolverDialog.java @@ -0,0 +1,125 @@ +package org.jabref.gui.collab; + +import java.util.ArrayList; +import java.util.List; + +import javafx.application.Platform; +import javafx.beans.binding.Bindings; +import javafx.beans.binding.BooleanBinding; +import javafx.collections.FXCollections; +import javafx.fxml.FXML; +import javafx.scene.control.Button; +import javafx.scene.control.SelectionMode; +import javafx.scene.control.TableColumn; +import javafx.scene.control.TableView; +import javafx.scene.layout.BorderPane; + +import org.jabref.gui.undo.NamedCompound; +import org.jabref.gui.util.BaseDialog; +import org.jabref.logic.l10n.Localization; +import org.jabref.model.database.BibDatabaseContext; + +import com.airhacks.afterburner.views.ViewLoader; +import com.tobiasdiez.easybind.EasyBind; +import de.saxsys.mvvmfx.utils.commands.CommandBase; +import org.controlsfx.control.MasterDetailPane; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class ExternalChangesResolverDialog extends BaseDialog { + private final static Logger LOGGER = LoggerFactory.getLogger(ExternalChangesResolverDialog.class); + + @FXML + public TableView changesTableView; + @FXML + public Button openAdvancedMergeDialogButton; + @FXML + public MasterDetailPane materDetailPane; + @FXML + public BorderPane changeInfoPane; + @FXML + private TableColumn changeName; + + private final BibDatabaseContext database; + private final NamedCompound ce = new NamedCompound(Localization.lang("Merged external changes")); + + private final List acceptedChanges = new ArrayList<>(); + + private final BooleanBinding areChangesResolved; + + private final BooleanBinding canOpenAdvancedMergeDialog; + + public ExternalChangesResolverDialog(BibDatabaseContext database, List changes) { + this.database = database; + + this.setTitle(Localization.lang("External changes")); + this.getDialogPane().setPrefSize(800, 600); + + ViewLoader.view(this) + .load().setAsDialogPane(this); + changesTableView.itemsProperty().set(FXCollections.observableList(changes)); + changesTableView.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE); + changesTableView.getSelectionModel().selectFirst(); + + areChangesResolved = Bindings.createBooleanBinding(() -> changesTableView.getItems().isEmpty(), changesTableView.itemsProperty().getValue()); + EasyBind.subscribe(areChangesResolved, isResolved -> { + if (isResolved) { + LOGGER.info("Closing ExternalChangesResolverDialog"); + close(); + } + }); + + canOpenAdvancedMergeDialog = Bindings.createBooleanBinding(() -> changesTableView.getSelectionModel().getSelectedItems().size() == 1 && + changesTableView.getSelectionModel().getSelectedItems().get(0).hasAdvancedMergeDialog(), changesTableView.getSelectionModel().getSelectedItems()); + + openAdvancedMergeDialogButton.disableProperty().bind(canOpenAdvancedMergeDialog.not()); + + setResultConverter(button -> { + if (areChangesResolved.get()) { + LOGGER.info("External changes are resolved successfully"); + Platform.runLater(this::applyChanges); + // TODO: panel.getUndoManager().addEdit(ce); + return true; + } else { + LOGGER.info("External changes ARE NOT resolved"); + return false; + } + }); + EasyBind.subscribe(changesTableView.getSelectionModel().selectedItemProperty(), selectedChange -> { + if (selectedChange != null) { + changeInfoPane.setCenter(selectedChange.description()); + } + }); + } + + @FXML + private void initialize() { + changeName.setCellValueFactory(data -> data.getValue().nameProperty()); + } + + @FXML + public void denyChanges() { + changesTableView.getItems().removeAll(changesTableView.getSelectionModel().getSelectedItems()); + } + + @FXML + public void acceptChanges() { + // Changes will be applied when closing the dialog as a transaction + acceptedChanges.addAll(changesTableView.getSelectionModel().getSelectedItems()); + changesTableView.getItems().removeAll(changesTableView.getSelectionModel().getSelectedItems()); + } + + @FXML + public void openAdvancedMergeDialog() { + assert changesTableView.getSelectionModel().getSelectedItems().size() == 1; + assert changesTableView.getSelectionModel().getSelectedItems().get(0).hasAdvancedMergeDialog(); + + DatabaseChangeViewModel changeViewModel = changesTableView.getSelectionModel().getSelectedItems().get(0); + changeViewModel.openAdvancedMergeDialog().ifPresent(CommandBase::execute); + } + + private void applyChanges() { + acceptedChanges.forEach(change -> change.makeChange(database, ce)); + ce.end(); + } +} diff --git a/src/main/java/org/jabref/gui/collab/ExternalChangesResolverViewModel.java b/src/main/java/org/jabref/gui/collab/ExternalChangesResolverViewModel.java new file mode 100644 index 00000000000..1453a9cf033 --- /dev/null +++ b/src/main/java/org/jabref/gui/collab/ExternalChangesResolverViewModel.java @@ -0,0 +1,6 @@ +package org.jabref.gui.collab; + +import org.jabref.gui.AbstractViewModel; + +public class ExternalChangesResolverViewModel extends AbstractViewModel { +} diff --git a/src/main/java/org/jabref/gui/collab/MetaDataChangeViewModel.java b/src/main/java/org/jabref/gui/collab/MetaDataChangeViewModel.java index 83b6351fd78..a60bccaa6e9 100644 --- a/src/main/java/org/jabref/gui/collab/MetaDataChangeViewModel.java +++ b/src/main/java/org/jabref/gui/collab/MetaDataChangeViewModel.java @@ -39,5 +39,8 @@ public Node description() { @Override public void makeChange(BibDatabaseContext database, NamedCompound undoEdit) { database.setMetaData(metaDataDiff.getNewMetaData()); + // group change is handled by GroupChangeViewModel, so we set the groups root to the original value + // to prevent any inconsistency + metaDataDiff.getGroupDifferences().ifPresent(groupDiff -> database.getMetaData().setGroups(groupDiff.getOriginalGroupRoot())); } } diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AbstractAutomaticFieldEditorTabViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AbstractAutomaticFieldEditorTabViewModel.java new file mode 100644 index 00000000000..fe6e60fee01 --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AbstractAutomaticFieldEditorTabViewModel.java @@ -0,0 +1,48 @@ +package org.jabref.gui.edit.automaticfiededitor; + +import java.util.Collection; +import java.util.Comparator; +import java.util.EnumSet; +import java.util.HashSet; +import java.util.Objects; +import java.util.Set; + +import javafx.collections.FXCollections; +import javafx.collections.ObservableList; + +import org.jabref.gui.AbstractViewModel; +import org.jabref.gui.StateManager; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.StandardField; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public abstract class AbstractAutomaticFieldEditorTabViewModel extends AbstractViewModel { + public static final Logger LOGGER = LoggerFactory.getLogger(AbstractAutomaticFieldEditorTabViewModel.class); + + protected final StateManager stateManager; + + private final ObservableList allFields = FXCollections.observableArrayList(); + + public AbstractAutomaticFieldEditorTabViewModel(BibDatabase bibDatabase, StateManager stateManager) { + Objects.requireNonNull(bibDatabase); + Objects.requireNonNull(stateManager); + this.stateManager = stateManager; + + addFields(EnumSet.allOf(StandardField.class)); + addFields(bibDatabase.getAllVisibleFields()); + allFields.sort(Comparator.comparing(Field::getName)); + } + + public ObservableList getAllFields() { + return allFields; + } + + private void addFields(Collection extends Field> fields) { + Set fieldsSet = new HashSet<>(allFields); + fieldsSet.addAll(fields); + allFields.setAll(fieldsSet); + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorAction.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorAction.java index a56af9dfa78..904762ca02a 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorAction.java +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorAction.java @@ -20,7 +20,6 @@ public AutomaticFieldEditorAction(StateManager stateManager, DialogService dialo @Override public void execute() { - dialogService.showCustomDialogAndWait(new AutomaticFieldEditorDialog(stateManager.getSelectedEntries(), - stateManager.getActiveDatabase().orElseThrow())); + dialogService.showCustomDialogAndWait(new AutomaticFieldEditorDialog(stateManager)); } } diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.fxml b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.fxml index f79462369a1..47d8338dca4 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.fxml +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.fxml @@ -10,6 +10,5 @@ - + diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.java index 5ecbf06b275..9f6a4663d9d 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.java +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorDialog.java @@ -1,39 +1,50 @@ package org.jabref.gui.edit.automaticfiededitor; +import java.util.ArrayList; import java.util.List; +import javax.swing.undo.CannotUndoException; import javax.swing.undo.UndoManager; import javafx.fxml.FXML; import javafx.scene.control.ButtonBar; -import javafx.scene.control.ButtonType; import javafx.scene.control.Tab; import javafx.scene.control.TabPane; import org.jabref.gui.Globals; +import org.jabref.gui.StateManager; import org.jabref.gui.util.BaseDialog; -import org.jabref.gui.util.ControlHelper; import org.jabref.logic.l10n.Localization; -import org.jabref.model.database.BibDatabaseContext; +import org.jabref.model.database.BibDatabase; import org.jabref.model.entry.BibEntry; import com.airhacks.afterburner.views.ViewLoader; +import com.tobiasdiez.easybind.EasyBind; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class AutomaticFieldEditorDialog extends BaseDialog { + + private static final Logger LOGGER = LoggerFactory.getLogger(AutomaticFieldEditorDialog.class); -public class AutomaticFieldEditorDialog extends BaseDialog { - @FXML public ButtonType saveButton; - @FXML public ButtonType cancelButton; @FXML private TabPane tabPane; private final UndoManager undoManager; - private final BibDatabaseContext databaseContext; + private final BibDatabase database; private final List selectedEntries; + + private final StateManager stateManager; + private AutomaticFieldEditorViewModel viewModel; - public AutomaticFieldEditorDialog(List selectedEntries, BibDatabaseContext databaseContext) { - this.selectedEntries = selectedEntries; - this.databaseContext = databaseContext; + private List notificationPanes = new ArrayList<>(); + + public AutomaticFieldEditorDialog(StateManager stateManager) { + this.selectedEntries = stateManager.getSelectedEntries(); + this.database = stateManager.getActiveDatabase().orElseThrow().getDatabase(); + this.stateManager = stateManager; this.undoManager = Globals.undoManager; this.setTitle(Localization.lang("Automatic field editor")); @@ -42,8 +53,14 @@ public AutomaticFieldEditorDialog(List selectedEntries, BibDatabaseCon .load() .setAsDialogPane(this); - ControlHelper.setAction(saveButton, getDialogPane(), event -> saveChangesAndCloseDialog()); - ControlHelper.setAction(cancelButton, getDialogPane(), event -> cancelChangesAndCloseDialog()); + setResultConverter(buttonType -> { + if (buttonType != null && buttonType.getButtonData() == ButtonBar.ButtonData.OK_DONE) { + saveChanges(); + } else { + cancelChanges(); + } + return ""; + }); // This will prevent all dialog buttons from having the same size // Read more: https://stackoverflow.com/questions/45866249/javafx-8-alert-different-button-sizes @@ -54,20 +71,30 @@ public AutomaticFieldEditorDialog(List selectedEntries, BibDatabaseCon @FXML public void initialize() { - viewModel = new AutomaticFieldEditorViewModel(selectedEntries, databaseContext, undoManager); + viewModel = new AutomaticFieldEditorViewModel(selectedEntries, database, undoManager, stateManager); for (AutomaticFieldEditorTab tabModel : viewModel.getFieldEditorTabs()) { - tabPane.getTabs().add(new Tab(tabModel.getTabName(), tabModel.getContent())); + NotificationPaneAdapter notificationPane = new NotificationPaneAdapter(tabModel.getContent()); + notificationPanes.add(notificationPane); + tabPane.getTabs().add(new Tab(tabModel.getTabName(), notificationPane)); } + + EasyBind.listen(stateManager.lastAutomaticFieldEditorEditProperty(), (obs, old, lastEdit) -> { + viewModel.getDialogEdits().addEdit(lastEdit.getEdit()); + notificationPanes.get(lastEdit.getTabIndex()) + .notify(lastEdit.getAffectedEntries(), selectedEntries.size()); + }); } - private void saveChangesAndCloseDialog() { + private void saveChanges() { viewModel.saveChanges(); - close(); } - private void cancelChangesAndCloseDialog() { - viewModel.cancelChanges(); - close(); + private void cancelChanges() { + try { + viewModel.cancelChanges(); + } catch (CannotUndoException e) { + LOGGER.info("Could not undo", e); + } } } diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorViewModel.java index 0c334ddf6fa..3f517a1dabd 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorViewModel.java +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/AutomaticFieldEditorViewModel.java @@ -8,11 +8,12 @@ import javafx.collections.ObservableList; import org.jabref.gui.AbstractViewModel; -import org.jabref.gui.edit.automaticfiededitor.editfieldvalue.EditFieldValueTabView; +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.copyormovecontent.CopyOrMoveFieldContentTabView; +import org.jabref.gui.edit.automaticfiededitor.editfieldcontent.EditFieldContentTabView; import org.jabref.gui.edit.automaticfiededitor.renamefield.RenameFieldTabView; -import org.jabref.gui.edit.automaticfiededitor.twofields.TwoFieldsTabView; import org.jabref.gui.undo.NamedCompound; -import org.jabref.model.database.BibDatabaseContext; +import org.jabref.model.database.BibDatabase; import org.jabref.model.entry.BibEntry; public class AutomaticFieldEditorViewModel extends AbstractViewModel { @@ -20,17 +21,19 @@ public class AutomaticFieldEditorViewModel extends AbstractViewModel { private final ObservableList fieldEditorTabs = FXCollections.observableArrayList(); private final NamedCompound dialogEdits = new NamedCompound(NAMED_COMPOUND_EDITS); - private final BibDatabaseContext databaseContext; private final UndoManager undoManager; - public AutomaticFieldEditorViewModel(List selectedEntries, BibDatabaseContext databaseContext, UndoManager undoManager) { + public AutomaticFieldEditorViewModel(List selectedEntries, BibDatabase database, UndoManager undoManager, StateManager stateManager) { + this.undoManager = undoManager; fieldEditorTabs.addAll( - new EditFieldValueTabView(selectedEntries, databaseContext, dialogEdits), - new TwoFieldsTabView(selectedEntries, databaseContext, dialogEdits), - new RenameFieldTabView(selectedEntries, databaseContext, dialogEdits) + new EditFieldContentTabView(database, stateManager), + new CopyOrMoveFieldContentTabView(database, stateManager), + new RenameFieldTabView(database, stateManager) ); - this.databaseContext = databaseContext; - this.undoManager = undoManager; + } + + public NamedCompound getDialogEdits() { + return dialogEdits; } public ObservableList getFieldEditorTabs() { diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/LastAutomaticFieldEditorEdit.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/LastAutomaticFieldEditorEdit.java new file mode 100644 index 00000000000..a1263576608 --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/LastAutomaticFieldEditorEdit.java @@ -0,0 +1,44 @@ +package org.jabref.gui.edit.automaticfiededitor; + +import javax.swing.undo.AbstractUndoableEdit; +import javax.swing.undo.CannotRedoException; +import javax.swing.undo.CannotUndoException; + +import org.jabref.gui.undo.NamedCompound; + +public class LastAutomaticFieldEditorEdit extends AbstractUndoableEdit { + private final Integer affectedEntries; + private final NamedCompound edit; + + private final Integer tabIndex; + + public LastAutomaticFieldEditorEdit(Integer affectedEntries, Integer tabIndex, NamedCompound edit) { + this.affectedEntries = affectedEntries; + this.edit = edit; + this.tabIndex = tabIndex; + } + + public Integer getAffectedEntries() { + return affectedEntries; + } + + public NamedCompound getEdit() { + return edit; + } + + public Integer getTabIndex() { + return tabIndex; + } + + @Override + public void undo() throws CannotUndoException { + super.undo(); + edit.undo(); + } + + @Override + public void redo() throws CannotRedoException { + super.redo(); + edit.redo(); + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/MoveFieldValueAction.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/MoveFieldValueAction.java index 8863d17de78..660c2f8454b 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/MoveFieldValueAction.java +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/MoveFieldValueAction.java @@ -16,27 +16,48 @@ public class MoveFieldValueAction extends SimpleCommand { private final NamedCompound edits; - public MoveFieldValueAction(Field fromField, Field toField, List entries, NamedCompound edits) { + private int affectedEntriesCount; + + private final boolean overwriteToFieldContent; + + public MoveFieldValueAction(Field fromField, Field toField, List entries, NamedCompound edits, boolean overwriteToFieldContent) { this.fromField = fromField; this.toField = toField; this.entries = entries; this.edits = edits; + this.overwriteToFieldContent = overwriteToFieldContent; + } + + public MoveFieldValueAction(Field fromField, Field toField, List entries, NamedCompound edits) { + this(fromField, toField, entries, edits, true); } @Override public void execute() { + affectedEntriesCount = 0; for (BibEntry entry : entries) { String fromFieldValue = entry.getField(fromField).orElse(""); String toFieldValue = entry.getField(toField).orElse(""); - if (StringUtil.isNotBlank(fromFieldValue)) { - entry.setField(toField, fromFieldValue); - entry.setField(fromField, ""); - - edits.addEdit(new UndoableFieldChange(entry, fromField, fromFieldValue, null)); - edits.addEdit(new UndoableFieldChange(entry, toField, toFieldValue, fromFieldValue)); + if (overwriteToFieldContent || toFieldValue.isEmpty()) { + entry.setField(toField, fromFieldValue); + entry.setField(fromField, ""); + + edits.addEdit(new UndoableFieldChange(entry, fromField, fromFieldValue, null)); + edits.addEdit(new UndoableFieldChange(entry, toField, toFieldValue, fromFieldValue)); + affectedEntriesCount++; + } } } + edits.end(); } + + /** + * @return the number of affected entries + * */ + public int executeAndGetAffectedEntriesCount() { + execute(); + return affectedEntriesCount; + } } diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/NotificationPaneAdapter.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/NotificationPaneAdapter.java new file mode 100644 index 00000000000..4fc8b9d8b63 --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/NotificationPaneAdapter.java @@ -0,0 +1,23 @@ +package org.jabref.gui.edit.automaticfiededitor; + +import java.util.Collections; + +import javafx.scene.Node; +import javafx.util.Duration; + +import org.jabref.gui.LibraryTab; +import org.jabref.gui.icon.IconTheme; + +public class NotificationPaneAdapter extends LibraryTab.DatabaseNotification { + + public NotificationPaneAdapter(Node content) { + super(content); + } + + public void notify(int affectedEntries, int totalEntries) { + String notificationMessage = String.format("%d/%d affected entries", affectedEntries, totalEntries); + Node notificationGraphic = IconTheme.JabRefIcons.INTEGRITY_INFO.getGraphicNode(); + + notify(notificationGraphic, notificationMessage, Collections.emptyList(), Duration.seconds(2)); + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsTab.fxml b/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTab.fxml similarity index 71% rename from src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsTab.fxml rename to src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTab.fxml index 7ac475e6cd7..462dcd0dfda 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsTab.fxml +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTab.fxml @@ -11,7 +11,7 @@ - + @@ -30,19 +30,23 @@ - - - + + + - - - + + + - + diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTabView.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTabView.java new file mode 100644 index 00000000000..af742e708c4 --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTabView.java @@ -0,0 +1,109 @@ +package org.jabref.gui.edit.automaticfiededitor.copyormovecontent; + +import java.util.ArrayList; +import java.util.List; + +import javafx.application.Platform; +import javafx.fxml.FXML; +import javafx.scene.control.Button; +import javafx.scene.control.CheckBox; +import javafx.scene.control.ComboBox; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabView; +import org.jabref.gui.edit.automaticfiededitor.AutomaticFieldEditorTab; +import org.jabref.logic.l10n.Localization; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; + +import com.airhacks.afterburner.views.ViewLoader; +import com.tobiasdiez.easybind.EasyBind; +import de.saxsys.mvvmfx.utils.validation.visualization.ControlsFxVisualizer; + +import static org.jabref.gui.customentrytypes.CustomEntryTypeDialogViewModel.FIELD_STRING_CONVERTER; + +public class CopyOrMoveFieldContentTabView extends AbstractAutomaticFieldEditorTabView implements AutomaticFieldEditorTab { + public Button copyContentButton; + @FXML + private Button moveContentButton; + + @FXML + private Button swapContentButton; + + @FXML + private ComboBox fromFieldComboBox; + @FXML + private ComboBox toFieldComboBox; + + @FXML + private CheckBox overwriteFieldContentCheckBox; + + private CopyOrMoveFieldContentTabViewModel viewModel; + private final List selectedEntries; + private final BibDatabase database; + private final StateManager stateManager; + + private final ControlsFxVisualizer visualizer = new ControlsFxVisualizer(); + + public CopyOrMoveFieldContentTabView(BibDatabase database, StateManager stateManager) { + this.selectedEntries = new ArrayList<>(stateManager.getSelectedEntries()); + this.database = database; + this.stateManager = stateManager; + + ViewLoader.view(this) + .root(this) + .load(); + } + + public void initialize() { + viewModel = new CopyOrMoveFieldContentTabViewModel(selectedEntries, database, stateManager); + initializeFromAndToComboBox(); + + viewModel.overwriteFieldContentProperty().bindBidirectional(overwriteFieldContentCheckBox.selectedProperty()); + + moveContentButton.disableProperty().bind(viewModel.canMoveProperty().not()); + swapContentButton.disableProperty().bind(viewModel.canSwapProperty().not()); + copyContentButton.disableProperty().bind(viewModel.toFieldValidationStatus().validProperty().not()); + overwriteFieldContentCheckBox.disableProperty().bind(viewModel.toFieldValidationStatus().validProperty().not()); + + Platform.runLater(() -> { + visualizer.initVisualization(viewModel.toFieldValidationStatus(), toFieldComboBox, true); + }); + } + + private void initializeFromAndToComboBox() { + fromFieldComboBox.getItems().setAll(viewModel.getAllFields()); + toFieldComboBox.getItems().setAll(viewModel.getAllFields()); + + fromFieldComboBox.setConverter(FIELD_STRING_CONVERTER); + + toFieldComboBox.setConverter(FIELD_STRING_CONVERTER); + + fromFieldComboBox.valueProperty().bindBidirectional(viewModel.fromFieldProperty()); + toFieldComboBox.valueProperty().bindBidirectional(viewModel.toFieldProperty()); + + EasyBind.listen(fromFieldComboBox.getEditor().textProperty(), observable -> fromFieldComboBox.commitValue()); + EasyBind.listen(toFieldComboBox.getEditor().textProperty(), observable -> toFieldComboBox.commitValue()); + } + + @Override + public String getTabName() { + return Localization.lang("Copy or Move content"); + } + + @FXML + void copyContent() { + viewModel.copyValue(); + } + + @FXML + void moveContent() { + viewModel.moveValue(); + } + + @FXML + void swapContent() { + viewModel.swapValues(); + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTabViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTabViewModel.java new file mode 100644 index 00000000000..3ed3a1b38ee --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/copyormovecontent/CopyOrMoveFieldContentTabViewModel.java @@ -0,0 +1,183 @@ +package org.jabref.gui.edit.automaticfiededitor.copyormovecontent; + +import java.util.ArrayList; +import java.util.List; + +import javafx.beans.binding.BooleanBinding; +import javafx.beans.property.BooleanProperty; +import javafx.beans.property.ObjectProperty; +import javafx.beans.property.SimpleBooleanProperty; +import javafx.beans.property.SimpleObjectProperty; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabViewModel; +import org.jabref.gui.edit.automaticfiededitor.LastAutomaticFieldEditorEdit; +import org.jabref.gui.edit.automaticfiededitor.MoveFieldValueAction; +import org.jabref.gui.undo.NamedCompound; +import org.jabref.gui.undo.UndoableFieldChange; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.strings.StringUtil; + +import de.saxsys.mvvmfx.utils.validation.FunctionBasedValidator; +import de.saxsys.mvvmfx.utils.validation.ValidationMessage; +import de.saxsys.mvvmfx.utils.validation.ValidationStatus; +import de.saxsys.mvvmfx.utils.validation.Validator; + +public class CopyOrMoveFieldContentTabViewModel extends AbstractAutomaticFieldEditorTabViewModel { + public static final int TAB_INDEX = 1; + private final ObjectProperty fromField = new SimpleObjectProperty<>(StandardField.ABSTRACT); + + private final ObjectProperty toField = new SimpleObjectProperty<>(StandardField.AUTHOR); + + private final BooleanProperty overwriteFieldContent = new SimpleBooleanProperty(Boolean.FALSE); + private final List selectedEntries; + + private final Validator toFieldValidator; + + private final BooleanBinding canMove; + + private final BooleanBinding canSwap; + + public CopyOrMoveFieldContentTabViewModel(List selectedEntries, BibDatabase bibDatabase, StateManager stateManager) { + super(bibDatabase, stateManager); + this.selectedEntries = new ArrayList<>(selectedEntries); + + toFieldValidator = new FunctionBasedValidator<>(toField, field -> { + if (StringUtil.isBlank(field.getName())) { + return ValidationMessage.error("Field name cannot be empty"); + } else if (StringUtil.containsWhitespace(field.getName())) { + return ValidationMessage.error("Field name cannot have whitespace characters"); + } + return null; + }); + + canMove = BooleanBinding.booleanExpression(toFieldValidationStatus().validProperty()) + .and(overwriteFieldContentProperty()); + + canSwap = BooleanBinding.booleanExpression(toFieldValidationStatus().validProperty()) + .and(overwriteFieldContentProperty()); + } + + public ValidationStatus toFieldValidationStatus() { + return toFieldValidator.getValidationStatus(); + } + + public BooleanBinding canMoveProperty() { + return canMove; + } + + public BooleanBinding canSwapProperty() { + return canSwap; + } + + public Field getFromField() { + return fromField.get(); + } + + public ObjectProperty fromFieldProperty() { + return fromField; + } + + public Field getToField() { + return toField.get(); + } + + public ObjectProperty toFieldProperty() { + return toField; + } + + public boolean isOverwriteFieldContent() { + return overwriteFieldContent.get(); + } + + public BooleanProperty overwriteFieldContentProperty() { + return overwriteFieldContent; + } + + public void copyValue() { + NamedCompound copyFieldValueEdit = new NamedCompound("COPY_FIELD_VALUE"); + int affectedEntriesCount = 0; + for (BibEntry entry : selectedEntries) { + String fromFieldValue = entry.getField(fromField.get()).orElse(""); + String toFieldValue = entry.getField(toField.get()).orElse(""); + + if (overwriteFieldContent.get() || StringUtil.isBlank(toFieldValue)) { + if (StringUtil.isNotBlank(fromFieldValue)) { + entry.setField(toField.get(), fromFieldValue); + copyFieldValueEdit.addEdit(new UndoableFieldChange(entry, + toField.get(), + toFieldValue, + fromFieldValue)); + affectedEntriesCount++; + } + } + } + if (copyFieldValueEdit.hasEdits()) { + copyFieldValueEdit.end(); + } + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, TAB_INDEX, copyFieldValueEdit + )); + } + + public void moveValue() { + NamedCompound moveEdit = new NamedCompound("MOVE_EDIT"); + int affectedEntriesCount = 0; + if (overwriteFieldContent.get()) { + affectedEntriesCount = new MoveFieldValueAction(fromField.get(), + toField.get(), + selectedEntries, + moveEdit).executeAndGetAffectedEntriesCount(); + + if (moveEdit.hasEdits()) { + moveEdit.end(); + } + } + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, TAB_INDEX, moveEdit + )); + } + + public void swapValues() { + NamedCompound swapFieldValuesEdit = new NamedCompound("SWAP_FIELD_VALUES"); + int affectedEntriesCount = 0; + for (BibEntry entry : selectedEntries) { + String fromFieldValue = entry.getField(fromField.get()).orElse(""); + String toFieldValue = entry.getField(toField.get()).orElse(""); + + if (overwriteFieldContent.get() && StringUtil.isNotBlank(fromFieldValue) && StringUtil.isNotBlank(toFieldValue)) { + entry.setField(toField.get(), fromFieldValue); + entry.setField(fromField.get(), toFieldValue); + + swapFieldValuesEdit.addEdit(new UndoableFieldChange( + entry, + toField.get(), + toFieldValue, + fromFieldValue + )); + + swapFieldValuesEdit.addEdit(new UndoableFieldChange( + entry, + fromField.get(), + fromFieldValue, + toFieldValue + )); + affectedEntriesCount++; + } + } + + if (swapFieldValuesEdit.hasEdits()) { + swapFieldValuesEdit.end(); + } + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, TAB_INDEX, swapFieldValuesEdit + )); + } + + public List getSelectedEntries() { + return selectedEntries; + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueTab.fxml b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentTab.fxml similarity index 85% rename from src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueTab.fxml rename to src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentTab.fxml index 4cf8a44016e..1beac33dd48 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueTab.fxml +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentTab.fxml @@ -15,7 +15,7 @@ - + @@ -32,7 +32,8 @@ - + @@ -42,7 +43,7 @@ - + @@ -51,13 +52,13 @@ - - + + - + diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentTabView.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentTabView.java new file mode 100644 index 00000000000..af1b4f9d9c1 --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentTabView.java @@ -0,0 +1,100 @@ +package org.jabref.gui.edit.automaticfiededitor.editfieldcontent; + +import java.util.List; + +import javafx.application.Platform; +import javafx.fxml.FXML; +import javafx.scene.control.Button; +import javafx.scene.control.CheckBox; +import javafx.scene.control.ComboBox; +import javafx.scene.control.TextField; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabView; +import org.jabref.logic.l10n.Localization; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; + +import com.airhacks.afterburner.views.ViewLoader; +import com.tobiasdiez.easybind.EasyBind; +import de.saxsys.mvvmfx.utils.validation.visualization.ControlsFxVisualizer; + +import static org.jabref.gui.customentrytypes.CustomEntryTypeDialogViewModel.FIELD_STRING_CONVERTER; + +public class EditFieldContentTabView extends AbstractAutomaticFieldEditorTabView { + public Button appendValueButton; + public Button clearFieldButton; + public Button setValueButton; + @FXML + private ComboBox fieldComboBox; + + @FXML + private TextField fieldValueTextField; + + @FXML + private CheckBox overwriteFieldContentCheckBox; + + private final List selectedEntries; + private final BibDatabase database; + + private EditFieldContentViewModel viewModel; + + private final StateManager stateManager; + + private final ControlsFxVisualizer visualizer = new ControlsFxVisualizer(); + + public EditFieldContentTabView(BibDatabase database, StateManager stateManager) { + this.selectedEntries = stateManager.getSelectedEntries(); + this.database = database; + this.stateManager = stateManager; + + ViewLoader.view(this) + .root(this) + .load(); + } + + @FXML + public void initialize() { + viewModel = new EditFieldContentViewModel(database, selectedEntries, stateManager); + fieldComboBox.setConverter(FIELD_STRING_CONVERTER); + + fieldComboBox.getItems().setAll(viewModel.getAllFields()); + + fieldComboBox.getSelectionModel().selectFirst(); + + fieldComboBox.valueProperty().bindBidirectional(viewModel.selectedFieldProperty()); + EasyBind.listen(fieldComboBox.getEditor().textProperty(), observable -> fieldComboBox.commitValue()); + + fieldValueTextField.textProperty().bindBidirectional(viewModel.fieldValueProperty()); + + overwriteFieldContentCheckBox.selectedProperty().bindBidirectional(viewModel.overwriteFieldContentProperty()); + + appendValueButton.disableProperty().bind(viewModel.canAppendProperty().not()); + setValueButton.disableProperty().bind(viewModel.fieldValidationStatus().validProperty().not()); + clearFieldButton.disableProperty().bind(viewModel.fieldValidationStatus().validProperty().not()); + overwriteFieldContentCheckBox.disableProperty().bind(viewModel.fieldValidationStatus().validProperty().not()); + + Platform.runLater(() -> visualizer.initVisualization(viewModel.fieldValidationStatus(), fieldComboBox, true)); + } + + @Override + public String getTabName() { + return Localization.lang("Edit content"); + } + + @FXML + void appendToFieldValue() { + viewModel.appendToFieldValue(); + } + + @FXML + void clearField() { + viewModel.clearSelectedField(); + } + + @FXML + void setFieldValue() { + viewModel.setFieldValue(); + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentViewModel.java new file mode 100644 index 00000000000..933e1d4f87b --- /dev/null +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldcontent/EditFieldContentViewModel.java @@ -0,0 +1,164 @@ +package org.jabref.gui.edit.automaticfiededitor.editfieldcontent; + +import java.util.ArrayList; +import java.util.List; +import java.util.Optional; + +import javafx.beans.binding.Bindings; +import javafx.beans.binding.BooleanBinding; +import javafx.beans.property.BooleanProperty; +import javafx.beans.property.ObjectProperty; +import javafx.beans.property.SimpleBooleanProperty; +import javafx.beans.property.SimpleObjectProperty; +import javafx.beans.property.SimpleStringProperty; +import javafx.beans.property.StringProperty; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabViewModel; +import org.jabref.gui.edit.automaticfiededitor.LastAutomaticFieldEditorEdit; +import org.jabref.gui.undo.NamedCompound; +import org.jabref.gui.undo.UndoableFieldChange; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.strings.StringUtil; + +import de.saxsys.mvvmfx.utils.validation.FunctionBasedValidator; +import de.saxsys.mvvmfx.utils.validation.ValidationMessage; +import de.saxsys.mvvmfx.utils.validation.ValidationStatus; +import de.saxsys.mvvmfx.utils.validation.Validator; + +public class EditFieldContentViewModel extends AbstractAutomaticFieldEditorTabViewModel { + public static final int TAB_INDEX = 0; + + private final List selectedEntries; + + private final StringProperty fieldValue = new SimpleStringProperty(""); + + private final ObjectProperty selectedField = new SimpleObjectProperty<>(StandardField.AUTHOR); + + private final BooleanProperty overwriteFieldContent = new SimpleBooleanProperty(Boolean.FALSE); + + private final Validator fieldValidator; + private final BooleanBinding canAppend; + + public EditFieldContentViewModel(BibDatabase database, List selectedEntries, StateManager stateManager) { + super(database, stateManager); + this.selectedEntries = new ArrayList<>(selectedEntries); + + fieldValidator = new FunctionBasedValidator<>(selectedField, field -> { + if (StringUtil.isBlank(field.getName())) { + return ValidationMessage.error("Field name cannot be empty"); + } else if (StringUtil.containsWhitespace(field.getName())) { + return ValidationMessage.error("Field name cannot have whitespace characters"); + } + return null; + }); + + canAppend = Bindings.and(overwriteFieldContentProperty(), fieldValidationStatus().validProperty()); + } + + public ValidationStatus fieldValidationStatus() { + return fieldValidator.getValidationStatus(); + } + + public BooleanBinding canAppendProperty() { + return canAppend; + } + + public void clearSelectedField() { + NamedCompound clearFieldEdit = new NamedCompound("CLEAR_SELECTED_FIELD"); + int affectedEntriesCount = 0; + for (BibEntry entry : selectedEntries) { + Optional oldFieldValue = entry.getField(selectedField.get()); + if (oldFieldValue.isPresent()) { + entry.clearField(selectedField.get()) + .ifPresent(fieldChange -> clearFieldEdit.addEdit(new UndoableFieldChange(fieldChange))); + affectedEntriesCount++; + } + } + + if (clearFieldEdit.hasEdits()) { + clearFieldEdit.end(); + } + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, + TAB_INDEX, + clearFieldEdit + )); + } + + public void setFieldValue() { + NamedCompound setFieldEdit = new NamedCompound("CHANGE_SELECTED_FIELD"); + String toSetFieldValue = fieldValue.getValue(); + int affectedEntriesCount = 0; + for (BibEntry entry : selectedEntries) { + Optional oldFieldValue = entry.getField(selectedField.get()); + if (oldFieldValue.isEmpty() || overwriteFieldContent.get()) { + entry.setField(selectedField.get(), toSetFieldValue) + .ifPresent(fieldChange -> setFieldEdit.addEdit(new UndoableFieldChange(fieldChange))); + fieldValue.set(""); + // TODO: increment affected entries only when UndoableFieldChange.isPresent() + affectedEntriesCount++; + } + } + + if (setFieldEdit.hasEdits()) { + setFieldEdit.end(); + } + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, + TAB_INDEX, + setFieldEdit + )); + } + + public void appendToFieldValue() { + NamedCompound appendToFieldEdit = new NamedCompound("APPEND_TO_SELECTED_FIELD"); + String toAppendFieldValue = fieldValue.getValue(); + int affectedEntriesCount = 0; + for (BibEntry entry : selectedEntries) { + Optional oldFieldValue = entry.getField(selectedField.get()); + // Append button should be disabled if 'overwriteNonEmptyFields' is false + if (overwriteFieldContent.get()) { + String newFieldValue = oldFieldValue.orElse("").concat(toAppendFieldValue); + + entry.setField(selectedField.get(), newFieldValue) + .ifPresent(fieldChange -> appendToFieldEdit.addEdit(new UndoableFieldChange(fieldChange))); + + fieldValue.set(""); + affectedEntriesCount++; + } + } + + if (appendToFieldEdit.hasEdits()) { + appendToFieldEdit.end(); + } + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, + TAB_INDEX, + appendToFieldEdit + )); + } + + public ObjectProperty selectedFieldProperty() { + return selectedField; + } + + public Field getSelectedField() { + return selectedFieldProperty().get(); + } + + public String getFieldValue() { + return fieldValue.get(); + } + + public StringProperty fieldValueProperty() { + return fieldValue; + } + + public BooleanProperty overwriteFieldContentProperty() { + return overwriteFieldContent; + } +} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueTabView.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueTabView.java deleted file mode 100644 index 3bd9ea57a09..00000000000 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueTabView.java +++ /dev/null @@ -1,95 +0,0 @@ -package org.jabref.gui.edit.automaticfiededitor.editfieldvalue; - -import java.util.Comparator; -import java.util.List; - -import javafx.fxml.FXML; -import javafx.scene.control.Button; -import javafx.scene.control.CheckBox; -import javafx.scene.control.ComboBox; -import javafx.scene.control.TextField; -import javafx.util.StringConverter; - -import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabView; -import org.jabref.gui.undo.NamedCompound; -import org.jabref.logic.l10n.Localization; -import org.jabref.model.database.BibDatabaseContext; -import org.jabref.model.entry.BibEntry; -import org.jabref.model.entry.field.Field; -import org.jabref.model.entry.field.FieldFactory; - -import com.airhacks.afterburner.views.ViewLoader; - -public class EditFieldValueTabView extends AbstractAutomaticFieldEditorTabView { - public Button appendValueButton; - @FXML - private ComboBox fieldComboBox; - - @FXML - private TextField fieldValueTextField; - - @FXML - private CheckBox overwriteNonEmptyFieldsCheckBox; - - private final List selectedEntries; - private final BibDatabaseContext databaseContext; - - private EditFieldValueViewModel viewModel; - - private final NamedCompound dialogEdits; - - public EditFieldValueTabView(List selectedEntries, BibDatabaseContext databaseContext, NamedCompound dialogEdits) { - this.selectedEntries = selectedEntries; - this.databaseContext = databaseContext; - this.dialogEdits = dialogEdits; - - ViewLoader.view(this) - .root(this) - .load(); - } - - @FXML - public void initialize() { - viewModel = new EditFieldValueViewModel(databaseContext, selectedEntries, dialogEdits); - fieldComboBox.setConverter(new StringConverter<>() { - @Override - public String toString(Field field) { - return field.getName(); - } - - @Override - public Field fromString(String name) { - return FieldFactory.parseField(name); - } - }); - fieldComboBox.getItems().addAll(viewModel.getAllFields().sorted(Comparator.comparing(Field::getName))); - fieldComboBox.getSelectionModel().selectFirst(); - viewModel.selectedFieldProperty().bindBidirectional(fieldComboBox.valueProperty()); - - viewModel.fieldValueProperty().bindBidirectional(fieldValueTextField.textProperty()); - - viewModel.overwriteNonEmptyFieldsProperty().bindBidirectional(overwriteNonEmptyFieldsCheckBox.selectedProperty()); - - appendValueButton.disableProperty().bind(overwriteNonEmptyFieldsCheckBox.selectedProperty().not()); - } - - @Override - public String getTabName() { - return Localization.lang("Edit field value"); - } - - @FXML - void appendToFieldValue() { - viewModel.appendToFieldValue(); - } - - @FXML - void clearField() { - viewModel.clearSelectedField(); - } - - @FXML - void setFieldValue() { - viewModel.setFieldValue(); - } -} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueViewModel.java deleted file mode 100644 index d11928ebc53..00000000000 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/editfieldvalue/EditFieldValueViewModel.java +++ /dev/null @@ -1,135 +0,0 @@ -package org.jabref.gui.edit.automaticfiededitor.editfieldvalue; - -import java.util.ArrayList; -import java.util.List; -import java.util.Optional; - -import javafx.beans.property.BooleanProperty; -import javafx.beans.property.ObjectProperty; -import javafx.beans.property.SimpleBooleanProperty; -import javafx.beans.property.SimpleObjectProperty; -import javafx.beans.property.SimpleStringProperty; -import javafx.beans.property.StringProperty; -import javafx.collections.FXCollections; -import javafx.collections.ObservableList; - -import org.jabref.gui.AbstractViewModel; -import org.jabref.gui.undo.NamedCompound; -import org.jabref.gui.undo.UndoableFieldChange; -import org.jabref.model.database.BibDatabaseContext; -import org.jabref.model.entry.BibEntry; -import org.jabref.model.entry.field.Field; - -public class EditFieldValueViewModel extends AbstractViewModel { - private final BibDatabaseContext databaseContext; - private final List selectedEntries; - - private final StringProperty fieldValue = new SimpleStringProperty(); - - private final ObjectProperty selectedField = new SimpleObjectProperty<>(); - - private final BooleanProperty overwriteNonEmptyFields = new SimpleBooleanProperty(); - - private final ObservableList allFields = FXCollections.observableArrayList(); - - private final NamedCompound dialogEdits; - - public EditFieldValueViewModel(BibDatabaseContext databaseContext, List selectedEntries, NamedCompound dialogEdits) { - this.databaseContext = databaseContext; - this.selectedEntries = new ArrayList<>(selectedEntries); - this.dialogEdits = dialogEdits; - - allFields.addAll(databaseContext.getDatabase().getAllVisibleFields().stream().toList()); - } - - public void clearSelectedField() { - NamedCompound clearFieldEdit = new NamedCompound("CLEAR_SELECTED_FIELD"); - - for (BibEntry entry : selectedEntries) { - Optional oldFieldValue = entry.getField(selectedField.get()); - if (oldFieldValue.isPresent()) { - entry.setField(selectedField.get(), ""); - - clearFieldEdit.addEdit(new UndoableFieldChange(entry, - selectedField.get(), - oldFieldValue.orElse(null), - fieldValue.get())); - } - } - - if (clearFieldEdit.hasEdits()) { - clearFieldEdit.end(); - dialogEdits.addEdit(clearFieldEdit); - } - } - - public void setFieldValue() { - NamedCompound setFieldEdit = new NamedCompound("CHANGE_SELECTED_FIELD"); - String toSetFieldValue = fieldValue.getValue(); - - for (BibEntry entry : selectedEntries) { - Optional oldFieldValue = entry.getField(selectedField.get()); - if (oldFieldValue.isEmpty() || overwriteNonEmptyFields.get()) { - entry.setField(selectedField.get(), toSetFieldValue); - - setFieldEdit.addEdit(new UndoableFieldChange(entry, - selectedField.get(), - oldFieldValue.orElse(null), - toSetFieldValue)); - fieldValue.set(""); - } - } - - if (setFieldEdit.hasEdits()) { - setFieldEdit.end(); - dialogEdits.addEdit(setFieldEdit); - } - } - - public void appendToFieldValue() { - NamedCompound appendToFieldEdit = new NamedCompound("APPEND_TO_SELECTED_FIELD"); - String toAppendFieldValue = fieldValue.getValue(); - - for (BibEntry entry : selectedEntries) { - Optional oldFieldValue = entry.getField(selectedField.get()); - // Append button should be disabled if 'overwriteNonEmptyFields' is false - if (overwriteNonEmptyFields.get()) { - String newFieldValue = oldFieldValue.orElse("").concat(toAppendFieldValue); - - entry.setField(selectedField.get(), newFieldValue); - appendToFieldEdit.addEdit(new UndoableFieldChange(entry, - selectedField.get(), - oldFieldValue.orElse(null), - newFieldValue) - ); - - fieldValue.set(""); - } - } - - if (appendToFieldEdit.hasEdits()) { - appendToFieldEdit.end(); - dialogEdits.addEdit(appendToFieldEdit); - } - } - - public ObservableList getAllFields() { - return allFields; - } - - public ObjectProperty selectedFieldProperty() { - return selectedField; - } - - public String getFieldValue() { - return fieldValue.get(); - } - - public StringProperty fieldValueProperty() { - return fieldValue; - } - - public BooleanProperty overwriteNonEmptyFieldsProperty() { - return overwriteNonEmptyFields; - } -} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTab.fxml b/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTab.fxml index 388f1f567c0..9e4b5b54269 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTab.fxml +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTab.fxml @@ -46,7 +46,7 @@ - + diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTabView.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTabView.java index 8029cfc7827..f39ae2efc86 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTabView.java +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldTabView.java @@ -1,38 +1,45 @@ package org.jabref.gui.edit.automaticfiededitor.renamefield; -import java.util.Comparator; import java.util.List; +import javafx.application.Platform; import javafx.fxml.FXML; +import javafx.scene.control.Button; import javafx.scene.control.ComboBox; import javafx.scene.control.TextField; -import javafx.util.StringConverter; +import org.jabref.gui.StateManager; import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabView; import org.jabref.gui.edit.automaticfiededitor.AutomaticFieldEditorTab; -import org.jabref.gui.undo.NamedCompound; import org.jabref.logic.l10n.Localization; -import org.jabref.model.database.BibDatabaseContext; +import org.jabref.model.database.BibDatabase; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.field.Field; -import org.jabref.model.entry.field.FieldFactory; import com.airhacks.afterburner.views.ViewLoader; +import com.tobiasdiez.easybind.EasyBind; +import de.saxsys.mvvmfx.utils.validation.visualization.ControlsFxVisualizer; + +import static org.jabref.gui.customentrytypes.CustomEntryTypeDialogViewModel.FIELD_STRING_CONVERTER; public class RenameFieldTabView extends AbstractAutomaticFieldEditorTabView implements AutomaticFieldEditorTab { + @FXML + private Button renameButton; @FXML private ComboBox fieldComboBox; @FXML private TextField newFieldNameTextField; private final List selectedEntries; - private final BibDatabaseContext databaseContext; - private final NamedCompound dialogEdits; + private final BibDatabase database; + private final StateManager stateManager; private RenameFieldViewModel viewModel; - public RenameFieldTabView(List selectedEntries, BibDatabaseContext databaseContext, NamedCompound dialogEdits) { - this.selectedEntries = selectedEntries; - this.databaseContext = databaseContext; - this.dialogEdits = dialogEdits; + private final ControlsFxVisualizer visualizer = new ControlsFxVisualizer(); + + public RenameFieldTabView(BibDatabase database, StateManager stateManager) { + this.selectedEntries = stateManager.getSelectedEntries(); + this.database = database; + this.stateManager = stateManager; ViewLoader.view(this) .root(this) @@ -41,25 +48,23 @@ public RenameFieldTabView(List selectedEntries, BibDatabaseContext dat @FXML public void initialize() { - viewModel = new RenameFieldViewModel(selectedEntries, databaseContext, dialogEdits); - - fieldComboBox.getItems().addAll(viewModel.getAllFields().sorted(Comparator.comparing(Field::getName))); - fieldComboBox.setConverter(new StringConverter<>() { - @Override - public String toString(Field field) { - return field.getName(); - } - - @Override - public Field fromString(String name) { - return FieldFactory.parseField(name); - } - }); + viewModel = new RenameFieldViewModel(selectedEntries, database, stateManager); + fieldComboBox.getItems().setAll(viewModel.getAllFields()); fieldComboBox.getSelectionModel().selectFirst(); - viewModel.selectedFieldProperty().bindBidirectional(fieldComboBox.valueProperty()); - viewModel.newFieldNameProperty().bindBidirectional(newFieldNameTextField.textProperty()); + fieldComboBox.setConverter(FIELD_STRING_CONVERTER); + + fieldComboBox.valueProperty().bindBidirectional(viewModel.selectedFieldProperty()); + EasyBind.listen(fieldComboBox.getEditor().textProperty(), observable -> fieldComboBox.commitValue()); + + renameButton.disableProperty().bind(viewModel.canRenameProperty().not()); + + newFieldNameTextField.textProperty().bindBidirectional(viewModel.newFieldNameProperty()); + + Platform.runLater(() -> { + visualizer.initVisualization(viewModel.fieldNameValidationStatus(), newFieldNameTextField, true); + }); } @Override diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldViewModel.java index f8e430ccc51..6f22623d466 100644 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldViewModel.java +++ b/src/main/java/org/jabref/gui/edit/automaticfiededitor/renamefield/RenameFieldViewModel.java @@ -2,37 +2,70 @@ import java.util.List; +import javafx.beans.binding.Bindings; +import javafx.beans.binding.BooleanBinding; import javafx.beans.property.ObjectProperty; import javafx.beans.property.SimpleObjectProperty; import javafx.beans.property.SimpleStringProperty; import javafx.beans.property.StringProperty; -import javafx.collections.FXCollections; -import javafx.collections.ObservableList; -import org.jabref.gui.AbstractViewModel; +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabViewModel; +import org.jabref.gui.edit.automaticfiededitor.LastAutomaticFieldEditorEdit; import org.jabref.gui.edit.automaticfiededitor.MoveFieldValueAction; import org.jabref.gui.undo.NamedCompound; -import org.jabref.model.database.BibDatabaseContext; +import org.jabref.model.database.BibDatabase; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.strings.StringUtil; + +import de.saxsys.mvvmfx.utils.validation.FunctionBasedValidator; +import de.saxsys.mvvmfx.utils.validation.ValidationMessage; +import de.saxsys.mvvmfx.utils.validation.ValidationStatus; +import de.saxsys.mvvmfx.utils.validation.Validator; + +public class RenameFieldViewModel extends AbstractAutomaticFieldEditorTabViewModel { + public static final int TAB_INDEX = 2; + private final StringProperty newFieldName = new SimpleStringProperty(""); + private final ObjectProperty selectedField = new SimpleObjectProperty<>(StandardField.AUTHOR); + private final List selectedEntries; -public class RenameFieldViewModel extends AbstractViewModel { + private final Validator fieldValidator; - private final StringProperty newFieldName = new SimpleStringProperty(); - private final ObjectProperty selectedField = new SimpleObjectProperty<>(); + private final Validator fieldNameValidator; - private final ObservableList allFields = FXCollections.observableArrayList(); - private final List selectedEntries; - private final BibDatabaseContext databaseContext; - private final NamedCompound dialogEdits; + private final BooleanBinding canRename; - public RenameFieldViewModel(List selectedEntries, BibDatabaseContext databaseContext, NamedCompound dialogEdits) { + public RenameFieldViewModel(List selectedEntries, BibDatabase database, StateManager stateManager) { + super(database, stateManager); this.selectedEntries = selectedEntries; - this.databaseContext = databaseContext; - this.dialogEdits = dialogEdits; - allFields.addAll(databaseContext.getDatabase().getAllVisibleFields()); + fieldValidator = new FunctionBasedValidator<>(selectedField, field -> StringUtil.isNotBlank(field.getName()), + ValidationMessage.error("Field cannot be empty")); + fieldNameValidator = new FunctionBasedValidator<>(newFieldName, fieldName -> { + if (StringUtil.isBlank(fieldName)) { + return ValidationMessage.error("Field name cannot be empty"); + } else if (StringUtil.containsWhitespace(fieldName)) { + return ValidationMessage.error("Field name cannot have whitespace characters"); + } + return null; + }); + + canRename = Bindings.and(fieldValidationStatus().validProperty(), fieldNameValidationStatus().validProperty()); + } + + public ValidationStatus fieldValidationStatus() { + return fieldValidator.getValidationStatus(); + } + + public ValidationStatus fieldNameValidationStatus() { + return fieldNameValidator.getValidationStatus(); + } + + public BooleanBinding canRenameProperty() { + return canRename; } public String getNewFieldName() { @@ -43,6 +76,10 @@ public StringProperty newFieldNameProperty() { return newFieldName; } + public void setNewFieldName(String newName) { + newFieldNameProperty().set(newName); + } + public Field getSelectedField() { return selectedField.get(); } @@ -51,21 +88,27 @@ public ObjectProperty selectedFieldProperty() { return selectedField; } - public ObservableList getAllFields() { - return allFields; + public void selectField(Field field) { + selectedFieldProperty().set(field); } public void renameField() { NamedCompound renameEdit = new NamedCompound("RENAME_EDIT"); - - new MoveFieldValueAction(selectedField.get(), - FieldFactory.parseField(newFieldName.get()), - selectedEntries, - renameEdit).execute(); - - if (renameEdit.hasEdits()) { - renameEdit.end(); - dialogEdits.addEdit(renameEdit); + int affectedEntriesCount = 0; + if (fieldNameValidationStatus().isValid()) { + affectedEntriesCount = new MoveFieldValueAction(selectedField.get(), + FieldFactory.parseField(newFieldName.get()), + selectedEntries, + renameEdit, + false).executeAndGetAffectedEntriesCount(); + + if (renameEdit.hasEdits()) { + renameEdit.end(); + } } + + stateManager.setLastAutomaticFieldEditorEdit(new LastAutomaticFieldEditorEdit( + affectedEntriesCount, TAB_INDEX, renameEdit + )); } } diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsTabView.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsTabView.java deleted file mode 100644 index baf000ce3cb..00000000000 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsTabView.java +++ /dev/null @@ -1,118 +0,0 @@ -package org.jabref.gui.edit.automaticfiededitor.twofields; - -import java.util.ArrayList; -import java.util.Comparator; -import java.util.List; - -import javafx.fxml.FXML; -import javafx.scene.control.Button; -import javafx.scene.control.CheckBox; -import javafx.scene.control.ComboBox; -import javafx.util.StringConverter; - -import org.jabref.gui.edit.automaticfiededitor.AbstractAutomaticFieldEditorTabView; -import org.jabref.gui.edit.automaticfiededitor.AutomaticFieldEditorTab; -import org.jabref.gui.undo.NamedCompound; -import org.jabref.logic.l10n.Localization; -import org.jabref.model.database.BibDatabaseContext; -import org.jabref.model.entry.BibEntry; -import org.jabref.model.entry.field.Field; -import org.jabref.model.entry.field.FieldFactory; - -import com.airhacks.afterburner.views.ViewLoader; - -public class TwoFieldsTabView extends AbstractAutomaticFieldEditorTabView implements AutomaticFieldEditorTab { - @FXML - private Button moveValueButton; - - @FXML - private Button swapValuesButton; - - @FXML - private ComboBox fromFieldComboBox; - @FXML - private ComboBox toFieldComboBox; - - @FXML - private CheckBox overwriteNonEmptyFields; - - private TwoFieldsViewModel viewModel; - private final List selectedEntries; - private final BibDatabaseContext databaseContext; - private final NamedCompound dialogEdits; - - public TwoFieldsTabView(List selectedEntries, BibDatabaseContext databaseContext, NamedCompound dialogEdits) { - this.selectedEntries = new ArrayList<>(selectedEntries); - this.databaseContext = databaseContext; - this.dialogEdits = dialogEdits; - - ViewLoader.view(this) - .root(this) - .load(); - } - - public void initialize() { - viewModel = new TwoFieldsViewModel(selectedEntries, databaseContext, dialogEdits); - initializeFromAndToComboBox(); - - viewModel.overwriteNonEmptyFieldsProperty().bindBidirectional(overwriteNonEmptyFields.selectedProperty()); - - moveValueButton.disableProperty().bind(viewModel.overwriteNonEmptyFieldsProperty().not()); - swapValuesButton.disableProperty().bind(viewModel.overwriteNonEmptyFieldsProperty().not()); - } - - private void initializeFromAndToComboBox() { - fromFieldComboBox.getItems().addAll(viewModel.getAllFields().sorted(Comparator.comparing(Field::getName))); - toFieldComboBox.getItems().addAll(viewModel.getAllFields().sorted(Comparator.comparing(Field::getName))); - - fromFieldComboBox.setConverter(new StringConverter<>() { - @Override - public String toString(Field field) { - return field.getName(); - } - - @Override - public Field fromString(String name) { - return FieldFactory.parseField(name); - } - }); - - toFieldComboBox.setConverter(new StringConverter<>() { - @Override - public String toString(Field field) { - return field.getName(); - } - - @Override - public Field fromString(String name) { - return FieldFactory.parseField(name); - } - }); - - fromFieldComboBox.getSelectionModel().selectFirst(); - toFieldComboBox.getSelectionModel().selectLast(); - - viewModel.fromFieldProperty().bindBidirectional(fromFieldComboBox.valueProperty()); - viewModel.toFieldProperty().bindBidirectional(toFieldComboBox.valueProperty()); - } - - @Override - public String getTabName() { - return Localization.lang("Two fields"); - } - - @FXML - void copyValue() { - viewModel.copyValue(); - } - - @FXML - void moveValue() { - viewModel.moveValue(); - } - - @FXML - void swapValues() { - viewModel.swapValues(); - } -} diff --git a/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsViewModel.java b/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsViewModel.java deleted file mode 100644 index cf184630db9..00000000000 --- a/src/main/java/org/jabref/gui/edit/automaticfiededitor/twofields/TwoFieldsViewModel.java +++ /dev/null @@ -1,141 +0,0 @@ -package org.jabref.gui.edit.automaticfiededitor.twofields; - -import java.util.ArrayList; -import java.util.List; - -import javafx.beans.property.BooleanProperty; -import javafx.beans.property.ObjectProperty; -import javafx.beans.property.SimpleBooleanProperty; -import javafx.beans.property.SimpleObjectProperty; -import javafx.collections.FXCollections; -import javafx.collections.ObservableList; - -import org.jabref.gui.AbstractViewModel; -import org.jabref.gui.edit.automaticfiededitor.MoveFieldValueAction; -import org.jabref.gui.undo.NamedCompound; -import org.jabref.gui.undo.UndoableFieldChange; -import org.jabref.model.database.BibDatabaseContext; -import org.jabref.model.entry.BibEntry; -import org.jabref.model.entry.field.Field; -import org.jabref.model.strings.StringUtil; - -public class TwoFieldsViewModel extends AbstractViewModel { - private final ObjectProperty fromField = new SimpleObjectProperty<>(); - - private final ObjectProperty toField = new SimpleObjectProperty<>(); - - private final BooleanProperty overwriteNonEmptyFields = new SimpleBooleanProperty(); - private final ObservableList allFields = FXCollections.observableArrayList(); - // TODO: Create an abstraction where selectedEntries, databaseContext and dialogEdits dependencies are shared across - // all automatic field editors tab view models - private final List selectedEntries; - private final BibDatabaseContext databaseContext; - private final NamedCompound dialogEdits; - - public TwoFieldsViewModel(List selectedEntries, BibDatabaseContext databaseContext, NamedCompound dialogEdits) { - this.selectedEntries = new ArrayList<>(selectedEntries); - this.databaseContext = databaseContext; - this.dialogEdits = dialogEdits; - - allFields.addAll(databaseContext.getDatabase().getAllVisibleFields()); - } - - public ObservableList getAllFields() { - return allFields; - } - - public Field getFromField() { - return fromField.get(); - } - - public ObjectProperty fromFieldProperty() { - return fromField; - } - - public Field getToField() { - return toField.get(); - } - - public ObjectProperty toFieldProperty() { - return toField; - } - - public boolean getOverwriteNonEmptyFields() { - return overwriteNonEmptyFields.get(); - } - - public BooleanProperty overwriteNonEmptyFieldsProperty() { - return overwriteNonEmptyFields; - } - - public void copyValue() { - NamedCompound copyFieldValueEdit = new NamedCompound("COPY_FIELD_VALUE"); - - for (BibEntry entry : selectedEntries) { - String fromFieldValue = entry.getField(fromField.get()).orElse(""); - String toFieldValue = entry.getField(toField.get()).orElse(""); - - if (overwriteNonEmptyFields.get() || StringUtil.isBlank(toFieldValue)) { - entry.setField(toField.get(), fromFieldValue); - - copyFieldValueEdit.addEdit(new UndoableFieldChange(entry, - toField.get(), - toFieldValue, - fromFieldValue)); - } - } - - if (copyFieldValueEdit.hasEdits()) { - copyFieldValueEdit.end(); - dialogEdits.addEdit(copyFieldValueEdit); - } - } - - public void moveValue() { - NamedCompound moveEdit = new NamedCompound("MOVE_EDIT"); - if (overwriteNonEmptyFields.get()) { - new MoveFieldValueAction(fromField.get(), - toField.get(), - selectedEntries, - moveEdit).execute(); - - if (moveEdit.hasEdits()) { - moveEdit.end(); - dialogEdits.addEdit(moveEdit); - } - } - } - - public void swapValues() { - NamedCompound swapFieldValuesEdit = new NamedCompound("SWAP_FIELD_VALUES"); - - for (BibEntry entry : selectedEntries) { - String fromFieldValue = entry.getField(fromField.get()).orElse(""); - String toFieldValue = entry.getField(toField.get()).orElse(""); - - if (overwriteNonEmptyFields.get() && StringUtil.isNotBlank(fromFieldValue) && StringUtil.isNotBlank(toFieldValue)) { - entry.setField(toField.get(), fromFieldValue); - entry.setField(fromField.get(), toFieldValue); - - swapFieldValuesEdit.addEdit(new UndoableFieldChange( - entry, - toField.get(), - toFieldValue, - fromFieldValue - )); - - swapFieldValuesEdit.addEdit(new UndoableFieldChange( - entry, - fromField.get(), - fromFieldValue, - toFieldValue - )); - } - } - - if (swapFieldValuesEdit.hasEdits()) { - swapFieldValuesEdit.end(); - dialogEdits.addEdit(swapFieldValuesEdit); - } - } -} diff --git a/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java b/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java index a5b513203a2..5066e762b1c 100644 --- a/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java +++ b/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java @@ -28,7 +28,6 @@ import javafx.scene.layout.BorderPane; import org.jabref.gui.DialogService; -import org.jabref.gui.Globals; import org.jabref.gui.LibraryTab; import org.jabref.gui.StateManager; import org.jabref.gui.citationkeypattern.GenerateCitationKeySingleAction; @@ -39,6 +38,7 @@ import org.jabref.gui.help.HelpAction; import org.jabref.gui.importer.GrobidOptInDialogHelper; import org.jabref.gui.keyboard.KeyBinding; +import org.jabref.gui.keyboard.KeyBindingRepository; import org.jabref.gui.menus.ChangeEntryTypeMenu; import org.jabref.gui.mergeentries.FetchAndMergeEntry; import org.jabref.gui.theme.ThemeManager; @@ -50,6 +50,7 @@ import org.jabref.logic.importer.EntryBasedFetcher; import org.jabref.logic.importer.WebFetchers; import org.jabref.logic.importer.fileformat.PdfMergeMetadataImporter; +import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.BibEntryTypesManager; @@ -107,6 +108,8 @@ public class EntryEditor extends BorderPane { @Inject private FileUpdateMonitor fileMonitor; @Inject private CountingUndoManager undoManager; @Inject private BibEntryTypesManager bibEntryTypesManager; + @Inject private KeyBindingRepository keyBindingRepository; + @Inject private JournalAbbreviationRepository journalAbbreviationRepository; private final List entryEditorTabs = new LinkedList<>(); @@ -173,7 +176,7 @@ public EntryEditor(LibraryTab libraryTab, ExternalFileTypes externalFileTypes) { */ private void setupKeyBindings() { this.addEventHandler(KeyEvent.KEY_PRESSED, event -> { - Optional keyBinding = Globals.getKeyPrefs().mapToKeyBinding(event); + Optional keyBinding = keyBindingRepository.mapToKeyBinding(event); if (keyBinding.isPresent()) { switch (keyBinding.get()) { case ENTRY_EDITOR_NEXT_PANEL: @@ -242,19 +245,19 @@ private List createTabs() { entryEditorTabs.add(new PreviewTab(databaseContext, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), ExternalFileTypes.getInstance())); // Required fields - entryEditorTabs.add(new RequiredFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, Globals.journalAbbreviationRepository)); + entryEditorTabs.add(new RequiredFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, journalAbbreviationRepository)); // Optional fields - entryEditorTabs.add(new OptionalFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, Globals.journalAbbreviationRepository)); - entryEditorTabs.add(new OptionalFields2Tab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, Globals.journalAbbreviationRepository)); - entryEditorTabs.add(new DeprecatedFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, Globals.journalAbbreviationRepository)); + entryEditorTabs.add(new OptionalFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, journalAbbreviationRepository)); + entryEditorTabs.add(new OptionalFields2Tab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, journalAbbreviationRepository)); + entryEditorTabs.add(new DeprecatedFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, journalAbbreviationRepository)); // Other fields - entryEditorTabs.add(new OtherFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, Globals.journalAbbreviationRepository)); + entryEditorTabs.add(new OtherFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, journalAbbreviationRepository)); // General fields from preferences for (Map.Entry> tab : entryEditorPreferences.getEntryEditorTabList().entrySet()) { - entryEditorTabs.add(new UserDefinedFieldsTab(tab.getKey(), tab.getValue(), databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, Globals.journalAbbreviationRepository)); + entryEditorTabs.add(new UserDefinedFieldsTab(tab.getKey(), tab.getValue(), databaseContext, libraryTab.getSuggestionProviders(), undoManager, dialogService, preferencesService, stateManager, themeManager, libraryTab.getIndexingTaskManager(), bibEntryTypesManager, ExternalFileTypes.getInstance(), taskExecutor, journalAbbreviationRepository)); } // Special tabs @@ -271,7 +274,7 @@ private List createTabs() { fileMonitor, dialogService, stateManager, - Globals.getKeyPrefs()); + keyBindingRepository); entryEditorTabs.add(sourceTab); // LaTeX citations tab @@ -361,7 +364,7 @@ private void setupToolBar() { typeLabel.setText(typedEntry.getTypeForDisplay()); // Add type change menu - ContextMenu typeMenu = new ChangeEntryTypeMenu().getChangeEntryTypePopupMenu(Collections.singletonList(entry), databaseContext, undoManager); + ContextMenu typeMenu = new ChangeEntryTypeMenu(Collections.singletonList(entry), databaseContext, undoManager, keyBindingRepository, bibEntryTypesManager).asContextMenu(); typeLabel.setOnMouseClicked(event -> typeMenu.show(typeLabel, Side.RIGHT, 0, 0)); typeChangeButton.setOnMouseClicked(event -> typeMenu.show(typeChangeButton, Side.RIGHT, 0, 0)); diff --git a/src/main/java/org/jabref/gui/entryeditor/SourceTab.java b/src/main/java/org/jabref/gui/entryeditor/SourceTab.java index d1773f9563f..1e3de019011 100644 --- a/src/main/java/org/jabref/gui/entryeditor/SourceTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/SourceTab.java @@ -194,8 +194,12 @@ private void setupSourceEditor() { sourceValidator.getValidationStatus().getMessages().addListener((InvalidationListener) c -> { ValidationStatus sourceValidationStatus = sourceValidator.getValidationStatus(); if (!sourceValidationStatus.isValid()) { - sourceValidationStatus.getHighestMessage().ifPresent(message -> - dialogService.showErrorDialogAndWait(message.getMessage())); + sourceValidationStatus.getHighestMessage().ifPresent(message -> { + String content = Localization.lang("User input via entry-editor in `{}bibtex source` tab led to failure.") + + "\n" + Localization.lang("Please check your library file for wrong syntax.") + + "\n\n" + message.getMessage(); + dialogService.showWarningDialogAndWait(Localization.lang("SourceTab error"), content); + }); } }); diff --git a/src/main/java/org/jabref/gui/importer/ImportAction.java b/src/main/java/org/jabref/gui/importer/ImportAction.java index aa63222685e..37566e8dc83 100644 --- a/src/main/java/org/jabref/gui/importer/ImportAction.java +++ b/src/main/java/org/jabref/gui/importer/ImportAction.java @@ -77,6 +77,10 @@ public void automatedImport(List filenames) { if (importError == null) { // TODO: No control flow using exceptions throw new JabRefException(Localization.lang("No entries found. Please make sure you are using the correct import filter.")); + } else if (importError instanceof ImportException) { + String content = Localization.lang("Please check your library file for wrong syntax.") + "\n\n" + + importError.getLocalizedMessage(); + DefaultTaskExecutor.runAndWaitInJavaFXThread(() -> dialogService.showWarningDialogAndWait(Localization.lang("Import error"), content)); } else { throw importError; } diff --git a/src/main/java/org/jabref/gui/importer/actions/OpenDatabaseAction.java b/src/main/java/org/jabref/gui/importer/actions/OpenDatabaseAction.java index 7c130a72f96..02c38ad9a34 100644 --- a/src/main/java/org/jabref/gui/importer/actions/OpenDatabaseAction.java +++ b/src/main/java/org/jabref/gui/importer/actions/OpenDatabaseAction.java @@ -24,6 +24,7 @@ import org.jabref.gui.shared.SharedDatabaseUIManager; import org.jabref.gui.theme.ThemeManager; import org.jabref.gui.util.BackgroundTask; +import org.jabref.gui.util.DefaultTaskExecutor; import org.jabref.gui.util.FileDialogConfiguration; import org.jabref.logic.autosaveandbackup.BackupManager; import org.jabref.logic.importer.OpenDatabase; @@ -200,6 +201,12 @@ private ParserResult loadDatabase(Path file) throws Exception { result = OpenDatabase.loadDatabase(fileToLoad, preferencesService.getImportFormatPreferences(), Globals.getFileUpdateMonitor()); + if (result.hasWarnings()) { + String content = Localization.lang("Please check your library file for wrong syntax.") + + "\n\n" + result.getErrorMessage(); + DefaultTaskExecutor.runInJavaFXThread(() -> + dialogService.showWarningDialogAndWait(Localization.lang("Open library error"), content)); + } } catch (IOException e) { result = ParserResult.fromError(e); LOGGER.error("Error opening file '{}'", fileToLoad, e); diff --git a/src/main/java/org/jabref/gui/maintable/MainTable.java b/src/main/java/org/jabref/gui/maintable/MainTable.java index 51494fe6315..32a80afb754 100644 --- a/src/main/java/org/jabref/gui/maintable/MainTable.java +++ b/src/main/java/org/jabref/gui/maintable/MainTable.java @@ -124,7 +124,8 @@ public MainTable(MainTableDataModel model, preferencesService, undoManager, Globals.getClipboardManager(), - Globals.TASK_EXECUTOR)) + Globals.TASK_EXECUTOR, + Globals.entryTypesManager)) .setOnDragDetected(this::handleOnDragDetected) .setOnDragDropped(this::handleOnDragDropped) .setOnDragOver(this::handleOnDragOver) @@ -314,6 +315,12 @@ public void paste() { } } + public void dropEntry(List entriesToAdd) { + for (BibEntry entry : entriesToAdd) { + importHandler.importEntryWithDuplicateCheck(database, (BibEntry) entry.clone()); + } + } + private void handleOnDragOver(TableRow row, BibEntryTableViewModel item, DragEvent event) { if (event.getDragboard().hasFiles()) { event.acceptTransferModes(TransferMode.ANY); @@ -350,8 +357,9 @@ private void handleOnDragDetected(TableRow row, BibEntry // The following is necesary to initiate the drag and drop in javafx, although we don't need the contents // It doesn't work without + // Drag'n'drop to other tabs use COPY TransferMode, drop to group sidepane use MOVE ClipboardContent content = new ClipboardContent(); - Dragboard dragboard = startDragAndDrop(TransferMode.MOVE); + Dragboard dragboard = startDragAndDrop(TransferMode.COPY_OR_MOVE); content.put(DragAndDropDataFormats.ENTRIES, ""); dragboard.setContent(content); diff --git a/src/main/java/org/jabref/gui/maintable/MainTableDataModel.java b/src/main/java/org/jabref/gui/maintable/MainTableDataModel.java index 5210613e4de..6ef5e500218 100644 --- a/src/main/java/org/jabref/gui/maintable/MainTableDataModel.java +++ b/src/main/java/org/jabref/gui/maintable/MainTableDataModel.java @@ -47,7 +47,10 @@ public MainTableDataModel(BibDatabaseContext context, PreferencesService prefere entriesFiltered = new FilteredList<>(entriesViewModel); entriesFiltered.predicateProperty().bind( - EasyBind.combine(stateManager.activeGroupProperty(), stateManager.activeSearchQueryProperty(), (groups, query) -> entry -> isMatched(groups, query, entry)) + EasyBind.combine(stateManager.activeGroupProperty(), + stateManager.activeSearchQueryProperty(), + groupsPreferences.groupViewModeProperty(), + (groups, query, groupViewMode) -> entry -> isMatched(groups, query, entry)) ); IntegerProperty resultSize = new SimpleIntegerProperty(); diff --git a/src/main/java/org/jabref/gui/maintable/RightClickMenu.java b/src/main/java/org/jabref/gui/maintable/RightClickMenu.java index 1d039fc5d37..15ff1e113e0 100644 --- a/src/main/java/org/jabref/gui/maintable/RightClickMenu.java +++ b/src/main/java/org/jabref/gui/maintable/RightClickMenu.java @@ -27,6 +27,7 @@ import org.jabref.gui.util.TaskExecutor; import org.jabref.logic.citationstyle.CitationStyleOutputFormat; import org.jabref.logic.citationstyle.CitationStylePreviewLayout; +import org.jabref.model.entry.BibEntryTypesManager; import org.jabref.model.entry.field.SpecialField; import org.jabref.preferences.PreferencesService; import org.jabref.preferences.PreviewPreferences; @@ -41,7 +42,8 @@ public static ContextMenu create(BibEntryTableViewModel entry, PreferencesService preferencesService, UndoManager undoManager, ClipBoardManager clipBoardManager, - TaskExecutor taskExecutor) { + TaskExecutor taskExecutor, + BibEntryTypesManager entryTypesManager) { ActionFactory factory = new ActionFactory(keyBindingRepository); ContextMenu contextMenu = new ContextMenu(); @@ -76,7 +78,7 @@ public static ContextMenu create(BibEntryTableViewModel entry, new SeparatorMenuItem(), - new ChangeEntryTypeMenu().getChangeEntryTypeMenu(libraryTab.getSelectedEntries(), libraryTab.getBibDatabaseContext(), libraryTab.getUndoManager()), + new ChangeEntryTypeMenu(libraryTab.getSelectedEntries(), libraryTab.getBibDatabaseContext(), libraryTab.getUndoManager(), keyBindingRepository, entryTypesManager).asSubMenu(), factory.createMenuItem(StandardActions.MERGE_WITH_FETCHED_ENTRY, new MergeWithFetchedEntryAction(libraryTab, dialogService, stateManager)) ); diff --git a/src/main/java/org/jabref/gui/menus/ChangeEntryTypeAction.java b/src/main/java/org/jabref/gui/menus/ChangeEntryTypeAction.java new file mode 100644 index 00000000000..97037881c5d --- /dev/null +++ b/src/main/java/org/jabref/gui/menus/ChangeEntryTypeAction.java @@ -0,0 +1,49 @@ +package org.jabref.gui.menus; + +import java.util.List; + +import javax.swing.undo.UndoManager; + +import javafx.beans.property.ReadOnlyStringProperty; +import javafx.beans.property.ReadOnlyStringWrapper; + +import org.jabref.gui.EntryTypeView; +import org.jabref.gui.actions.SimpleCommand; +import org.jabref.gui.undo.NamedCompound; +import org.jabref.gui.undo.UndoableChangeType; +import org.jabref.logic.l10n.Localization; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.types.EntryType; + +public class ChangeEntryTypeAction extends SimpleCommand { + + private final EntryType type; + private final List entries; + private final UndoManager undoManager; + private final ReadOnlyStringWrapper statusMessageProperty; + + public ChangeEntryTypeAction(EntryType type, List entries, UndoManager undoManager) { + this.type = type; + this.entries = entries; + this.undoManager = undoManager; + this.statusMessageProperty = new ReadOnlyStringWrapper(EntryTypeView.getDescription(type)); + } + + @Override + public void execute() { + NamedCompound compound = new NamedCompound(Localization.lang("Change entry type")); + entries.forEach(e -> e.setType(type) + .ifPresent(change -> compound.addEdit(new UndoableChangeType(change)))); + undoManager.addEdit(compound); + } + + @Override + public String getStatusMessage() { + return statusMessage.get(); + } + + @Override + public ReadOnlyStringProperty statusMessageProperty() { + return statusMessageProperty.getReadOnlyProperty(); + } +} diff --git a/src/main/java/org/jabref/gui/menus/ChangeEntryTypeMenu.java b/src/main/java/org/jabref/gui/menus/ChangeEntryTypeMenu.java index a57cb869e78..df9c8c4ef66 100644 --- a/src/main/java/org/jabref/gui/menus/ChangeEntryTypeMenu.java +++ b/src/main/java/org/jabref/gui/menus/ChangeEntryTypeMenu.java @@ -2,103 +2,111 @@ import java.util.Collection; import java.util.List; +import java.util.Optional; import javax.swing.undo.UndoManager; +import javafx.collections.FXCollections; import javafx.collections.ObservableList; import javafx.scene.control.ContextMenu; -import javafx.scene.control.CustomMenuItem; -import javafx.scene.control.Label; import javafx.scene.control.Menu; import javafx.scene.control.MenuItem; import javafx.scene.control.SeparatorMenuItem; -import javafx.scene.control.Tooltip; -import org.jabref.gui.EntryTypeView; -import org.jabref.gui.Globals; -import org.jabref.gui.undo.CountingUndoManager; -import org.jabref.gui.undo.NamedCompound; -import org.jabref.gui.undo.UndoableChangeType; +import org.jabref.gui.actions.ActionFactory; +import org.jabref.gui.keyboard.KeyBindingRepository; import org.jabref.logic.l10n.Localization; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.BibEntryType; +import org.jabref.model.entry.BibEntryTypesManager; import org.jabref.model.entry.types.BibtexEntryTypeDefinitions; -import org.jabref.model.entry.types.EntryType; import org.jabref.model.entry.types.IEEETranEntryTypeDefinitions; -import org.jabref.model.strings.StringUtil; public class ChangeEntryTypeMenu { - public ChangeEntryTypeMenu() { + private final List entries; + private final BibDatabaseContext bibDatabaseContext; + private final UndoManager undoManager; + private final ActionFactory factory; + private final BibEntryTypesManager entryTypesManager; + + public ChangeEntryTypeMenu(List entries, + BibDatabaseContext bibDatabaseContext, + UndoManager undoManager, + KeyBindingRepository keyBindingRepository, + BibEntryTypesManager entryTypesManager) { + this.entries = entries; + this.bibDatabaseContext = bibDatabaseContext; + this.undoManager = undoManager; + this.entryTypesManager = entryTypesManager; + this.factory = new ActionFactory(keyBindingRepository); } - public static MenuItem createMenuItem(EntryType type, List entries, UndoManager undoManager) { - CustomMenuItem menuItem = new CustomMenuItem(new Label(type.getDisplayName())); - menuItem.setOnAction(event -> { - NamedCompound compound = new NamedCompound(Localization.lang("Change entry type")); - entries.forEach(e -> e.setType(type) - .ifPresent(change -> compound.addEdit(new UndoableChangeType(change)))); - undoManager.addEdit(compound); - }); - String description = EntryTypeView.getDescription(type); - if (StringUtil.isNotBlank(description)) { - Tooltip tooltip = new Tooltip(description); - Tooltip.install(menuItem.getContent(), tooltip); - } - return menuItem; - } - - public ContextMenu getChangeEntryTypePopupMenu(List entries, BibDatabaseContext bibDatabaseContext, CountingUndoManager undoManager) { + public ContextMenu asContextMenu() { ContextMenu menu = new ContextMenu(); - populateComplete(menu.getItems(), entries, bibDatabaseContext, undoManager); + menu.getItems().setAll(getMenuItems(entries, bibDatabaseContext, undoManager)); return menu; } - public Menu getChangeEntryTypeMenu(List entries, BibDatabaseContext bibDatabaseContext, CountingUndoManager undoManager) { - Menu menu = new Menu(); - menu.setText(Localization.lang("Change entry type")); - populateComplete(menu.getItems(), entries, bibDatabaseContext, undoManager); + public Menu asSubMenu() { + Menu menu = new Menu(Localization.lang("Change entry type")); + menu.getItems().setAll(getMenuItems(entries, bibDatabaseContext, undoManager)); return menu; } - private void populateComplete(ObservableList items, List entries, BibDatabaseContext bibDatabaseContext, CountingUndoManager undoManager) { + private ObservableList getMenuItems(List entries, BibDatabaseContext bibDatabaseContext, UndoManager undoManager) { + ObservableList items = FXCollections.observableArrayList(); + if (bibDatabaseContext.isBiblatexMode()) { // Default BibLaTeX - populate(items, Globals.entryTypesManager.getAllTypes(BibDatabaseMode.BIBLATEX), entries, undoManager); + items.addAll(fromEntryTypes(entryTypesManager.getAllTypes(BibDatabaseMode.BIBLATEX), entries, undoManager)); // Custom types - populateSubMenu(items, Localization.lang("Custom"), Globals.entryTypesManager.getAllCustomTypes(BibDatabaseMode.BIBLATEX), entries, undoManager); + createSubMenu(Localization.lang("Custom"), entryTypesManager.getAllCustomTypes(BibDatabaseMode.BIBLATEX), entries, undoManager) + .ifPresent(subMenu -> + items.addAll(new SeparatorMenuItem(), + subMenu + )); } else { // Default BibTeX - populateSubMenu(items, BibDatabaseMode.BIBTEX.getFormattedName(), BibtexEntryTypeDefinitions.ALL, entries, undoManager); - items.remove(0); // Remove separator + createSubMenu(BibDatabaseMode.BIBTEX.getFormattedName(), BibtexEntryTypeDefinitions.ALL, entries, undoManager) + .ifPresent(items::add); // IEEETran - populateSubMenu(items, "IEEETran", IEEETranEntryTypeDefinitions.ALL, entries, undoManager); + createSubMenu("IEEETran", IEEETranEntryTypeDefinitions.ALL, entries, undoManager) + .ifPresent(subMenu -> items.addAll( + new SeparatorMenuItem(), + subMenu + )); // Custom types - populateSubMenu(items, Localization.lang("Custom"), Globals.entryTypesManager.getAllCustomTypes(BibDatabaseMode.BIBTEX), entries, undoManager); + createSubMenu(Localization.lang("Custom"), entryTypesManager.getAllCustomTypes(BibDatabaseMode.BIBTEX), entries, undoManager) + .ifPresent(subMenu -> items.addAll( + new SeparatorMenuItem(), + subMenu + )); } + + return items; } - private void populateSubMenu(ObservableList items, String text, List entryTypes, List entries, CountingUndoManager undoManager) { + private Optional createSubMenu(String text, List entryTypes, List entries, UndoManager undoManager) { + Menu subMenu = null; + if (!entryTypes.isEmpty()) { - items.add(new SeparatorMenuItem()); - Menu custom = new Menu(text); - populate(custom, entryTypes, entries, undoManager); - items.add(custom); + subMenu = factory.createMenu(() -> text); + subMenu.getItems().addAll(fromEntryTypes(entryTypes, entries, undoManager)); } - } - private void populate(ObservableList items, Collection types, List entries, UndoManager undoManager) { - for (BibEntryType type : types) { - items.add(createMenuItem(type.getType(), entries, undoManager)); - } + return Optional.ofNullable(subMenu); } - private void populate(Menu menu, Collection types, List entries, UndoManager undoManager) { - populate(menu.getItems(), types, entries, undoManager); + private List fromEntryTypes(Collection types, List entries, UndoManager undoManager) { + return types.stream() + .map(BibEntryType::getType) + .map(type -> factory.createMenuItem(type::getDisplayName, new ChangeEntryTypeAction(type, entries, undoManager))) + .toList(); } } diff --git a/src/main/java/org/jabref/gui/mergeentries/EntriesMergeResult.java b/src/main/java/org/jabref/gui/mergeentries/EntriesMergeResult.java new file mode 100644 index 00000000000..b134b11688a --- /dev/null +++ b/src/main/java/org/jabref/gui/mergeentries/EntriesMergeResult.java @@ -0,0 +1,8 @@ +package org.jabref.gui.mergeentries; + +import org.jabref.model.entry.BibEntry; + +public record EntriesMergeResult( + BibEntry originalLeftEntry, BibEntry originalRightEntry, BibEntry newLeftEntry, BibEntry newRightEntry, BibEntry mergedEntry +) { +} diff --git a/src/main/java/org/jabref/gui/mergeentries/FetchAndMergeEntry.java b/src/main/java/org/jabref/gui/mergeentries/FetchAndMergeEntry.java index 542788431cc..04eda76e077 100644 --- a/src/main/java/org/jabref/gui/mergeentries/FetchAndMergeEntry.java +++ b/src/main/java/org/jabref/gui/mergeentries/FetchAndMergeEntry.java @@ -91,7 +91,7 @@ private void showMergeDialog(BibEntry originalEntry, BibEntry fetchedEntry, WebF dialog.setTitle(Localization.lang("Merge entry with %0 information", fetcher.getName())); dialog.setLeftHeaderText(Localization.lang("Original entry")); dialog.setRightHeaderText(Localization.lang("Entry from %0", fetcher.getName())); - Optional mergedEntry = dialogService.showCustomDialogAndWait(dialog).map(MergeResult::mergedEntry); + Optional mergedEntry = dialogService.showCustomDialogAndWait(dialog).map(EntriesMergeResult::mergedEntry); if (mergedEntry.isPresent()) { NamedCompound ce = new NamedCompound(Localization.lang("Merge entry with %0 information", fetcher.getName())); diff --git a/src/main/java/org/jabref/gui/mergeentries/MergeEntriesAction.java b/src/main/java/org/jabref/gui/mergeentries/MergeEntriesAction.java index 7820cd95c92..82652a75fb4 100644 --- a/src/main/java/org/jabref/gui/mergeentries/MergeEntriesAction.java +++ b/src/main/java/org/jabref/gui/mergeentries/MergeEntriesAction.java @@ -1,21 +1,15 @@ package org.jabref.gui.mergeentries; -import java.util.Arrays; import java.util.List; import java.util.Optional; import org.jabref.gui.DialogService; -import org.jabref.gui.Globals; import org.jabref.gui.JabRefFrame; import org.jabref.gui.StateManager; import org.jabref.gui.actions.ActionHelper; import org.jabref.gui.actions.SimpleCommand; -import org.jabref.gui.undo.NamedCompound; -import org.jabref.gui.undo.UndoableInsertEntries; -import org.jabref.gui.undo.UndoableRemoveEntries; import org.jabref.logic.bibtex.comparator.EntryComparator; import org.jabref.logic.l10n.Localization; -import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.field.InternalField; @@ -38,7 +32,6 @@ public void execute() { if (stateManager.getActiveDatabase().isEmpty()) { return; } - BibDatabaseContext databaseContext = stateManager.getActiveDatabase().get(); // Check if there are two entries selected List selectedEntries = stateManager.getSelectedEntries(); @@ -68,20 +61,9 @@ public void execute() { MergeEntriesDialog dialog = new MergeEntriesDialog(first, second); dialog.setTitle(Localization.lang("Merge entries")); - Optional mergeResultOpt = dialogService.showCustomDialogAndWait(dialog); - mergeResultOpt.ifPresentOrElse(mergeResult -> { - // TODO: BibDatabase::insertEntry does not contain logic to mark the BasePanel as changed and to mark - // entries with a timestamp, only BasePanel::insertEntry does. Workaround for the moment is to get the - // BasePanel from the constructor injected JabRefFrame. Should be refactored and extracted! - frame.getCurrentLibraryTab().insertEntry(mergeResult.mergedEntry()); - - NamedCompound ce = new NamedCompound(Localization.lang("Merge entries")); - ce.addEdit(new UndoableInsertEntries(databaseContext.getDatabase(), mergeResult.mergedEntry())); - List entriesToRemove = Arrays.asList(one, two); - ce.addEdit(new UndoableRemoveEntries(databaseContext.getDatabase(), entriesToRemove)); - databaseContext.getDatabase().removeEntries(entriesToRemove); - ce.end(); - Globals.undoManager.addEdit(ce); + Optional mergeResultOpt = dialogService.showCustomDialogAndWait(dialog); + mergeResultOpt.ifPresentOrElse(entriesMergeResult -> { + new MergeTwoEntriesAction(entriesMergeResult, frame, stateManager).execute(); dialogService.notify(Localization.lang("Merged entries")); }, () -> dialogService.notify(Localization.lang("Canceled merging entries"))); diff --git a/src/main/java/org/jabref/gui/mergeentries/MergeEntriesDialog.java b/src/main/java/org/jabref/gui/mergeentries/MergeEntriesDialog.java index a93598494b0..4d9c06c4ef9 100644 --- a/src/main/java/org/jabref/gui/mergeentries/MergeEntriesDialog.java +++ b/src/main/java/org/jabref/gui/mergeentries/MergeEntriesDialog.java @@ -8,11 +8,15 @@ import org.jabref.logic.l10n.Localization; import org.jabref.model.entry.BibEntry; -public class MergeEntriesDialog extends BaseDialog { +public class MergeEntriesDialog extends BaseDialog { private final ThreeWayMergeView threeWayMergeView; + private final BibEntry one; + private final BibEntry two; public MergeEntriesDialog(BibEntry one, BibEntry two) { threeWayMergeView = new ThreeWayMergeView(one, two); + this.one = one; + this.two = two; init(); } @@ -31,7 +35,7 @@ private void init() { this.getDialogPane().getButtonTypes().setAll(ButtonType.CANCEL, replaceEntries); this.setResultConverter(buttonType -> { if (buttonType.equals(replaceEntries)) { - return new MergeResult(threeWayMergeView.getLeftEntry(), threeWayMergeView.getRightEntry(), threeWayMergeView.getMergedEntry()); + return new EntriesMergeResult(one, two, threeWayMergeView.getLeftEntry(), threeWayMergeView.getRightEntry(), threeWayMergeView.getMergedEntry()); } else { return null; } diff --git a/src/main/java/org/jabref/gui/mergeentries/MergeResult.java b/src/main/java/org/jabref/gui/mergeentries/MergeResult.java deleted file mode 100644 index 6db5a7a60e5..00000000000 --- a/src/main/java/org/jabref/gui/mergeentries/MergeResult.java +++ /dev/null @@ -1,8 +0,0 @@ -package org.jabref.gui.mergeentries; - -import org.jabref.model.entry.BibEntry; - -public record MergeResult( - BibEntry leftEntry, BibEntry rightEntry, BibEntry mergedEntry -) { -} diff --git a/src/main/java/org/jabref/gui/mergeentries/MergeTwoEntriesAction.java b/src/main/java/org/jabref/gui/mergeentries/MergeTwoEntriesAction.java new file mode 100644 index 00000000000..e494e1966b5 --- /dev/null +++ b/src/main/java/org/jabref/gui/mergeentries/MergeTwoEntriesAction.java @@ -0,0 +1,46 @@ +package org.jabref.gui.mergeentries; + +import java.util.Arrays; +import java.util.List; + +import org.jabref.gui.Globals; +import org.jabref.gui.JabRefFrame; +import org.jabref.gui.StateManager; +import org.jabref.gui.actions.SimpleCommand; +import org.jabref.gui.undo.NamedCompound; +import org.jabref.gui.undo.UndoableInsertEntries; +import org.jabref.gui.undo.UndoableRemoveEntries; +import org.jabref.logic.l10n.Localization; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; + +public class MergeTwoEntriesAction extends SimpleCommand { + private final EntriesMergeResult entriesMergeResult; + private final JabRefFrame frame; + private final StateManager stateManager; + + public MergeTwoEntriesAction(EntriesMergeResult entriesMergeResult, JabRefFrame frame, StateManager stateManager) { + this.entriesMergeResult = entriesMergeResult; + this.frame = frame; + this.stateManager = stateManager; + } + + @Override + public void execute() { + if (stateManager.getActiveDatabase().isEmpty()) { + return; + } + + BibDatabase database = stateManager.getActiveDatabase().get().getDatabase(); + + frame.getCurrentLibraryTab().insertEntry(entriesMergeResult.mergedEntry()); + + NamedCompound ce = new NamedCompound(Localization.lang("Merge entries")); + ce.addEdit(new UndoableInsertEntries(stateManager.getActiveDatabase().get().getDatabase(), entriesMergeResult.mergedEntry())); + List entriesToRemove = Arrays.asList(entriesMergeResult.originalLeftEntry(), entriesMergeResult.originalRightEntry()); + ce.addEdit(new UndoableRemoveEntries(database, entriesToRemove)); + database.removeEntries(entriesToRemove); + ce.end(); + Globals.undoManager.addEdit(ce); + } +} diff --git a/src/main/java/org/jabref/gui/openoffice/OpenOfficePanel.java b/src/main/java/org/jabref/gui/openoffice/OpenOfficePanel.java index 68f274e1a2d..298c0149309 100644 --- a/src/main/java/org/jabref/gui/openoffice/OpenOfficePanel.java +++ b/src/main/java/org/jabref/gui/openoffice/OpenOfficePanel.java @@ -381,6 +381,7 @@ protected OOBibBase call() throws Exception { connectTask.setOnFailed(value -> { Throwable ex = connectTask.getException(); + LOGGER.error("autodetect failed", ex); if (ex instanceof UnsatisfiedLinkError) { LOGGER.warn("Could not connect to running OpenOffice/LibreOffice", ex); @@ -395,6 +396,9 @@ protected OOBibBase call() throws Exception { + Localization.lang("Make sure you have installed OpenOffice/LibreOffice with Java support.") + "\n" + Localization.lang("If connecting manually, please verify program and library paths.") + "\n" + "\n" + Localization.lang("Error message:"), ex); + } else if (ex instanceof BootstrapException bootstrapEx) { + LOGGER.error("Exception boostrap cause", bootstrapEx.getTargetException()); + dialogService.showErrorDialogAndWait("Bootstrap error", bootstrapEx.getTargetException()); } else { dialogService.showErrorDialogAndWait(Localization.lang("Autodetection failed"), Localization.lang("Autodetection failed"), ex); } diff --git a/src/main/java/org/jabref/gui/search/RebuildFulltextSearchIndexAction.java b/src/main/java/org/jabref/gui/search/RebuildFulltextSearchIndexAction.java index 2de75512b39..90d4fe5f1bd 100644 --- a/src/main/java/org/jabref/gui/search/RebuildFulltextSearchIndexAction.java +++ b/src/main/java/org/jabref/gui/search/RebuildFulltextSearchIndexAction.java @@ -69,7 +69,7 @@ private void rebuildIndex() { } try { currentLibraryTab.get().getIndexingTaskManager().createIndex(PdfIndexer.of(databaseContext, filePreferences)); - currentLibraryTab.get().getIndexingTaskManager().addToIndex(PdfIndexer.of(databaseContext, filePreferences), databaseContext); + currentLibraryTab.get().getIndexingTaskManager().updateIndex(PdfIndexer.of(databaseContext, filePreferences), databaseContext); } catch (IOException e) { dialogService.notify(Localization.lang("Failed to access fulltext search index")); LOGGER.error("Failed to access fulltext search index", e); diff --git a/src/main/java/org/jabref/gui/shared/SharedDatabaseLoginDialogViewModel.java b/src/main/java/org/jabref/gui/shared/SharedDatabaseLoginDialogViewModel.java index 43bf6f86dac..970d36db98e 100644 --- a/src/main/java/org/jabref/gui/shared/SharedDatabaseLoginDialogViewModel.java +++ b/src/main/java/org/jabref/gui/shared/SharedDatabaseLoginDialogViewModel.java @@ -212,6 +212,9 @@ private void setPreferences() { } sharedDatabasePreferences.setRememberPassword(rememberPassword.get()); + + sharedDatabasePreferences.setFolder(folder.getValue()); + sharedDatabasePreferences.setAutosave(autosave.get()); } /** @@ -225,6 +228,8 @@ private void applyPreferences() { Optional sharedDatabaseUser = sharedDatabasePreferences.getUser(); Optional sharedDatabasePassword = sharedDatabasePreferences.getPassword(); boolean sharedDatabaseRememberPassword = sharedDatabasePreferences.getRememberPassword(); + Optional sharedDatabaseFolder = sharedDatabasePreferences.getFolder(); + boolean sharedDatabaseAutosave = sharedDatabasePreferences.getAutosave(); Optional sharedDatabaseKeystoreFile = sharedDatabasePreferences.getKeyStoreFile(); if (sharedDatabaseType.isPresent()) { @@ -248,6 +253,9 @@ private void applyPreferences() { } rememberPassword.set(sharedDatabaseRememberPassword); + + sharedDatabaseFolder.ifPresent(folder::set); + autosave.set(sharedDatabaseAutosave); } private boolean isSharedDatabaseAlreadyPresent(DBMSConnectionProperties connectionProperties) { diff --git a/src/main/java/org/jabref/gui/shared/SharedDatabaseUIManager.java b/src/main/java/org/jabref/gui/shared/SharedDatabaseUIManager.java index e5c2a0e9732..6b87d7c620a 100644 --- a/src/main/java/org/jabref/gui/shared/SharedDatabaseUIManager.java +++ b/src/main/java/org/jabref/gui/shared/SharedDatabaseUIManager.java @@ -15,8 +15,8 @@ import org.jabref.gui.LibraryTab; import org.jabref.gui.entryeditor.EntryEditor; import org.jabref.gui.exporter.SaveDatabaseAction; +import org.jabref.gui.mergeentries.EntriesMergeResult; import org.jabref.gui.mergeentries.MergeEntriesDialog; -import org.jabref.gui.mergeentries.MergeResult; import org.jabref.gui.undo.UndoableRemoveEntries; import org.jabref.logic.importer.ParserResult; import org.jabref.logic.l10n.Localization; @@ -100,7 +100,7 @@ public void listen(UpdateRefusedEvent updateRefusedEvent) { if (response.isPresent() && response.get().equals(merge)) { MergeEntriesDialog dialog = new MergeEntriesDialog(localBibEntry, sharedBibEntry); - Optional mergedEntry = dialogService.showCustomDialogAndWait(dialog).map(MergeResult::mergedEntry); + Optional mergedEntry = dialogService.showCustomDialogAndWait(dialog).map(EntriesMergeResult::mergedEntry); mergedEntry.ifPresent(mergedBibEntry -> { mergedBibEntry.getSharedBibEntryData().setSharedID(sharedBibEntry.getSharedBibEntryData().getSharedID()); diff --git a/src/main/java/org/jabref/logic/bibtex/comparator/GroupDiff.java b/src/main/java/org/jabref/logic/bibtex/comparator/GroupDiff.java index 4914ed9642d..35c2e5a1338 100644 --- a/src/main/java/org/jabref/logic/bibtex/comparator/GroupDiff.java +++ b/src/main/java/org/jabref/logic/bibtex/comparator/GroupDiff.java @@ -23,7 +23,7 @@ public static Optional compare(MetaData originalMetaData, MetaData ne final Optional newGroups = newMetaData.getGroups(); if (!originalGroups.equals(newGroups)) { - return Optional.of(new GroupDiff(newGroups.orElse(null), originalGroups.orElse(null))); + return Optional.of(new GroupDiff(originalGroups.orElse(null), newGroups.orElse(null))); } else { return Optional.empty(); } diff --git a/src/main/java/org/jabref/logic/bst/BibtexWidth.java b/src/main/java/org/jabref/logic/bst/BibtexWidth.java deleted file mode 100644 index 61e0b099f20..00000000000 --- a/src/main/java/org/jabref/logic/bst/BibtexWidth.java +++ /dev/null @@ -1,241 +0,0 @@ -package org.jabref.logic.bst; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -/** - * - * The |built_in| function {\.{purify\$}} pops the top (string) literal, removes - * nonalphanumeric characters except for |white_space| and |sep_char| characters - * (these get converted to a |space|) and removes certain alphabetic characters - * contained in the control sequences associated with a special character, and - * pushes the resulting string. If the literal isn't a string, it complains and - * pushes the null string. - * - */ -public class BibtexWidth { - - private static final Logger LOGGER = LoggerFactory.getLogger(BibtexWidth.class); - - /* - * Quoted from Bibtex: - * - * Now we initialize the system-dependent |char_width| array, for which - * |space| is the only |white_space| character given a nonzero printing - * width. The widths here are taken from Stanford's June~'87 $cmr10$~font - * and represent hundredths of a point (rounded), but since they're used - * only for relative comparisons, the units have no meaning. - */ - - private static int[] widths; - - static { - if (BibtexWidth.widths == null) { - BibtexWidth.widths = new int[128]; - - for (int i = 0; i < 128; i++) { - BibtexWidth.widths[i] = 0; - } - BibtexWidth.widths[32] = 278; - BibtexWidth.widths[33] = 278; - BibtexWidth.widths[34] = 500; - BibtexWidth.widths[35] = 833; - BibtexWidth.widths[36] = 500; - BibtexWidth.widths[37] = 833; - BibtexWidth.widths[38] = 778; - BibtexWidth.widths[39] = 278; - BibtexWidth.widths[40] = 389; - BibtexWidth.widths[41] = 389; - BibtexWidth.widths[42] = 500; - BibtexWidth.widths[43] = 778; - BibtexWidth.widths[44] = 278; - BibtexWidth.widths[45] = 333; - BibtexWidth.widths[46] = 278; - BibtexWidth.widths[47] = 500; - BibtexWidth.widths[48] = 500; - BibtexWidth.widths[49] = 500; - BibtexWidth.widths[50] = 500; - BibtexWidth.widths[51] = 500; - BibtexWidth.widths[52] = 500; - BibtexWidth.widths[53] = 500; - BibtexWidth.widths[54] = 500; - BibtexWidth.widths[55] = 500; - BibtexWidth.widths[56] = 500; - BibtexWidth.widths[57] = 500; - BibtexWidth.widths[58] = 278; - BibtexWidth.widths[59] = 278; - BibtexWidth.widths[60] = 278; - BibtexWidth.widths[61] = 778; - BibtexWidth.widths[62] = 472; - BibtexWidth.widths[63] = 472; - BibtexWidth.widths[64] = 778; - BibtexWidth.widths[65] = 750; - BibtexWidth.widths[66] = 708; - BibtexWidth.widths[67] = 722; - BibtexWidth.widths[68] = 764; - BibtexWidth.widths[69] = 681; - BibtexWidth.widths[70] = 653; - BibtexWidth.widths[71] = 785; - BibtexWidth.widths[72] = 750; - BibtexWidth.widths[73] = 361; - BibtexWidth.widths[74] = 514; - BibtexWidth.widths[75] = 778; - BibtexWidth.widths[76] = 625; - BibtexWidth.widths[77] = 917; - BibtexWidth.widths[78] = 750; - BibtexWidth.widths[79] = 778; - BibtexWidth.widths[80] = 681; - BibtexWidth.widths[81] = 778; - BibtexWidth.widths[82] = 736; - BibtexWidth.widths[83] = 556; - BibtexWidth.widths[84] = 722; - BibtexWidth.widths[85] = 750; - BibtexWidth.widths[86] = 750; - BibtexWidth.widths[87] = 1028; - BibtexWidth.widths[88] = 750; - BibtexWidth.widths[89] = 750; - BibtexWidth.widths[90] = 611; - BibtexWidth.widths[91] = 278; - BibtexWidth.widths[92] = 500; - BibtexWidth.widths[93] = 278; - BibtexWidth.widths[94] = 500; - BibtexWidth.widths[95] = 278; - BibtexWidth.widths[96] = 278; - BibtexWidth.widths[97] = 500; - BibtexWidth.widths[98] = 556; - BibtexWidth.widths[99] = 444; - BibtexWidth.widths[100] = 556; - BibtexWidth.widths[101] = 444; - BibtexWidth.widths[102] = 306; - BibtexWidth.widths[103] = 500; - BibtexWidth.widths[104] = 556; - BibtexWidth.widths[105] = 278; - BibtexWidth.widths[106] = 306; - BibtexWidth.widths[107] = 528; - BibtexWidth.widths[108] = 278; - BibtexWidth.widths[109] = 833; - BibtexWidth.widths[110] = 556; - BibtexWidth.widths[111] = 500; - BibtexWidth.widths[112] = 556; - BibtexWidth.widths[113] = 528; - BibtexWidth.widths[114] = 392; - BibtexWidth.widths[115] = 394; - BibtexWidth.widths[116] = 389; - BibtexWidth.widths[117] = 556; - BibtexWidth.widths[118] = 528; - BibtexWidth.widths[119] = 722; - BibtexWidth.widths[120] = 528; - BibtexWidth.widths[121] = 528; - BibtexWidth.widths[122] = 444; - BibtexWidth.widths[123] = 500; - BibtexWidth.widths[124] = 1000; - BibtexWidth.widths[125] = 500; - BibtexWidth.widths[126] = 500; - } - } - - private BibtexWidth() { - } - - private static int getSpecialCharWidth(char[] c, int pos) { - if ((pos + 1) < c.length) { - if ((c[pos] == 'o') && (c[pos + 1] == 'e')) { - return 778; - } - if ((c[pos] == 'O') && (c[pos + 1] == 'E')) { - return 1014; - } - if ((c[pos] == 'a') && (c[pos + 1] == 'e')) { - return 722; - } - if ((c[pos] == 'A') && (c[pos + 1] == 'E')) { - return 903; - } - if ((c[pos] == 's') && (c[pos + 1] == 's')) { - return 500; - } - } - return BibtexWidth.getCharWidth(c[pos]); - } - - public static int getCharWidth(char c) { - if ((c >= 0) && (c < 128)) { - return BibtexWidth.widths[c]; - } else { - return 0; - } - } - - public static int width(String toMeasure) { - /* - * From Bibtex: We use the natural width for all but special characters, - * and we complain if the string isn't brace-balanced. - */ - - int i = 0; - int n = toMeasure.length(); - int braceLevel = 0; - char[] c = toMeasure.toCharArray(); - int result = 0; - - /* - * From Bibtex: - * - * We use the natural widths of all characters except that some - * characters have no width: braces, control sequences (except for the - * usual 13 accented and foreign characters, whose widths are given in - * the next module), and |white_space| following control sequences (even - * a null control sequence). - * - */ - while (i < n) { - if (c[i] == '{') { - braceLevel++; - if ((braceLevel == 1) && ((i + 1) < n) && (c[i + 1] == '\\')) { - i++; // skip brace - while ((i < n) && (braceLevel > 0)) { - i++; // skip backslash - - int afterBackslash = i; - while ((i < n) && Character.isLetter(c[i])) { - i++; - } - if ((i < n) && (i == afterBackslash)) { - i++; // Skip non-alpha control seq - } else { - if (BibtexCaseChanger.findSpecialChar(c, afterBackslash).isPresent()) { - result += BibtexWidth.getSpecialCharWidth(c, afterBackslash); - } - } - while ((i < n) && Character.isWhitespace(c[i])) { - i++; - } - while ((i < n) && (braceLevel > 0) && (c[i] != '\\')) { - if (c[i] == '}') { - braceLevel--; - } else if (c[i] == '{') { - braceLevel++; - } else { - result += BibtexWidth.getCharWidth(c[i]); - } - i++; - } - } - continue; - } - } else if (c[i] == '}') { - if (braceLevel > 0) { - braceLevel--; - } else { - LOGGER.warn("Too many closing braces in string: " + toMeasure); - } - } - result += BibtexWidth.getCharWidth(c[i]); - i++; - } - if (braceLevel > 0) { - LOGGER.warn("No enough closing braces in string: " + toMeasure); - } - return result; - } -} diff --git a/src/main/java/org/jabref/logic/bst/BstEntry.java b/src/main/java/org/jabref/logic/bst/BstEntry.java new file mode 100644 index 00000000000..89571de8046 --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/BstEntry.java @@ -0,0 +1,21 @@ +package org.jabref.logic.bst; + +import java.util.HashMap; +import java.util.Map; + +import org.jabref.model.entry.BibEntry; + +public class BstEntry { + + public final BibEntry entry; + + public final Map localStrings = new HashMap<>(); + + public final Map fields = new HashMap<>(); + + public final Map localIntegers = new HashMap<>(); + + public BstEntry(BibEntry e) { + this.entry = e; + } +} diff --git a/src/main/java/org/jabref/logic/bst/BstFunctions.java b/src/main/java/org/jabref/logic/bst/BstFunctions.java new file mode 100644 index 00000000000..6689f83b2a5 --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/BstFunctions.java @@ -0,0 +1,931 @@ +package org.jabref.logic.bst; + +import java.util.HashMap; +import java.util.Map; +import java.util.Optional; +import java.util.Stack; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.jabref.logic.bst.util.BstCaseChanger; +import org.jabref.logic.bst.util.BstNameFormatter; +import org.jabref.logic.bst.util.BstPurifier; +import org.jabref.logic.bst.util.BstTextPrefixer; +import org.jabref.logic.bst.util.BstWidthCalculator; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.Author; +import org.jabref.model.entry.AuthorList; + +import org.antlr.v4.runtime.ParserRuleContext; +import org.antlr.v4.runtime.tree.ParseTree; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class BstFunctions { + private static final Logger LOGGER = LoggerFactory.getLogger(BstFunctions.class); + private static final Pattern ADD_PERIOD_PATTERN = Pattern.compile("([^.?!}\\s])(}|\\s)*$"); + + private final Map strings; + private final Map integers; + private final Map functions; + private final String preamble; + + private final Stack stack; + private final StringBuilder bbl; + + private int bstWarning = 0; + + @FunctionalInterface + public interface BstFunction { + + void execute(BstVMVisitor visitor, ParserRuleContext ctx); + + default void execute(BstVMVisitor visitor, ParserRuleContext ctx, BstEntry bstEntryContext) { + this.execute(visitor, ctx); + } + } + + public BstFunctions(BstVMContext bstVMContext, + StringBuilder bbl) { + this.strings = bstVMContext.strings(); + this.integers = bstVMContext.integers(); + this.functions = bstVMContext.functions(); + this.preamble = Optional.ofNullable(bstVMContext.bibDatabase()).flatMap(BibDatabase::getPreamble).orElse(""); + this.stack = bstVMContext.stack(); + + this.bbl = bbl; + } + + protected Map getBuiltInFunctions() { + Map builtInFunctions = new HashMap<>(); + + builtInFunctions.put(">", this::bstIsGreaterThan); + builtInFunctions.put("<", this::bstIsLowerThan); + builtInFunctions.put("=", this::bstEquals); + builtInFunctions.put("+", this::bstAdd); + builtInFunctions.put("-", this::bstSubtract); + builtInFunctions.put("*", this::bstConcat); + builtInFunctions.put(":=", new BstAssignFunction()); + builtInFunctions.put("add.period$", this::bstAddPeriod); + builtInFunctions.put("call.type$", new BstCallTypeFunction()); + builtInFunctions.put("change.case$", this::bstChangeCase); + builtInFunctions.put("chr.to.int$", this::bstChrToInt); + builtInFunctions.put("cite$", new BstCiteFunction()); + builtInFunctions.put("duplicate$", this::bstDuplicate); + builtInFunctions.put("empty$", this::bstEmpty); + builtInFunctions.put("format.name$", this::bstFormatName); + builtInFunctions.put("if$", this::bstIf); + builtInFunctions.put("int.to.chr$", this::bstIntToChr); + builtInFunctions.put("int.to.str$", this::bstIntToStr); + builtInFunctions.put("missing$", this::bstMissing); + builtInFunctions.put("newline$", this::bstNewLine); + builtInFunctions.put("num.names$", this::bstNumNames); + builtInFunctions.put("pop$", this::bstPop); + builtInFunctions.put("preamble$", this::bstPreamble); + builtInFunctions.put("purify$", this::bstPurify); + builtInFunctions.put("quote$", this::bstQuote); + builtInFunctions.put("skip$", this::bstSkip); + builtInFunctions.put("stack$", this::bstStack); + builtInFunctions.put("substring$", this::bstSubstring); + builtInFunctions.put("swap$", this::bstSwap); + builtInFunctions.put("text.length$", this::bstTextLength); + builtInFunctions.put("text.prefix$", this::bstTextPrefix); + builtInFunctions.put("top$", this::bstTop); + builtInFunctions.put("type$", new BstTypeFunction()); + builtInFunctions.put("warning$", this::bstWarning); + builtInFunctions.put("while$", this::bstWhile); + builtInFunctions.put("width$", this::bstWidth); + builtInFunctions.put("write$", this::bstWrite); + + return builtInFunctions; + } + + /** + * Pops the top two (integer) literals, compares them, and pushes + * the integer 1 if the second is greater than the first, 0 + * otherwise. + */ + private void bstIsGreaterThan(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation > (line %d)".formatted(ctx.start.getLine())); + } + Object o2 = stack.pop(); + Object o1 = stack.pop(); + + if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { + throw new BstVMException("Can only compare two integers with >"); + } + + stack.push(((Integer) o1).compareTo((Integer) o2) > 0 ? BstVM.TRUE : BstVM.FALSE); + } + + /** + * Pops the top two (integer) literals, compares them, and pushes + * the integer 1 if the second is lower than the first, 0 + * otherwise. + */ + private void bstIsLowerThan(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation <"); + } + Object o2 = stack.pop(); + Object o1 = stack.pop(); + + if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { + throw new BstVMException("Can only compare two integers with < (line %d)".formatted(ctx.start.getLine())); + } + + stack.push(((Integer) o1).compareTo((Integer) o2) < 0 ? BstVM.TRUE : BstVM.FALSE); + } + + /** + * Pops the top two (both integer or both string) literals, compares + * them, and pushes the integer 1 if they're equal, 0 otherwise. + */ + private void bstEquals(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation = (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + Object o2 = stack.pop(); + + if ((o1 == null) ^ (o2 == null)) { + stack.push(BstVM.FALSE); + return; + } + + if ((o1 == null) && (o2 == null)) { + stack.push(BstVM.TRUE); + return; + } + + stack.push(o1.equals(o2) ? BstVM.TRUE : BstVM.FALSE); + } + + /** + * Pops the top two (integer) literals and pushes their sum. + */ + private void bstAdd(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation + (line %d)".formatted(ctx.start.getLine())); + } + Object o2 = stack.pop(); + Object o1 = stack.pop(); + + if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { + throw new BstVMException("Can only compare two integers with + (line %d)".formatted(ctx.start.getLine())); + } + + stack.push((Integer) o1 + (Integer) o2); + } + + /** + * Pops the top two (integer) literals and pushes their difference + * (the first subtracted from the second). + */ + private void bstSubtract(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation - (line %d)".formatted(ctx.start.getLine())); + } + Object o2 = stack.pop(); + Object o1 = stack.pop(); + + if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { + throw new BstVMException("Can only subtract two integers with - (line %d)".formatted(ctx.start.getLine())); + } + + stack.push((Integer) o1 - (Integer) o2); + } + + /** + * Pops the top two (string) literals, concatenates them (in reverse + * order, that is, the order in which pushed), and pushes the + * resulting string. + */ + private void bstConcat(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation * (line %d)".formatted(ctx.start.getLine())); + } + Object o2 = stack.pop(); + Object o1 = stack.pop(); + + if (o1 == null) { + o1 = ""; + } + if (o2 == null) { + o2 = ""; + } + + if (!((o1 instanceof String) && (o2 instanceof String))) { + LOGGER.error("o1: {} ({})", o1, o1.getClass()); + LOGGER.error("o2: {} ({})", o2, o2.getClass()); + throw new BstVMException("Can only concatenate two String with * (line %d)".formatted(ctx.start.getLine())); + } + + stack.push(o1.toString() + o2); + } + + /** + * Pops the top two literals and assigns to the first (which must be + * a global or entry variable) the value of the second. + */ + public class BstAssignFunction implements BstFunction { + + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx) { + this.execute(visitor, ctx, null); + } + + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx, BstEntry bstEntry) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation := (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + Object o2 = stack.pop(); + + if (!(o1 instanceof BstVMVisitor.Identifier identifier)) { + throw new BstVMException("Invalid parameters (line %d)".formatted(ctx.start.getLine())); + } + String name = identifier.name(); + + if (o2 instanceof String value) { + if ((bstEntry != null) && bstEntry.localStrings.containsKey(name)) { + bstEntry.localStrings.put(name, value); + return; + } + + if (strings.containsKey(name)) { + strings.put(name, value); + } + } else if (o2 instanceof Integer value) { + if ((bstEntry != null) && bstEntry.localIntegers.containsKey(name)) { + bstEntry.localIntegers.put(name, value); + return; + } + + if (integers.containsKey(name)) { + integers.put(name, value); + } + } else { + throw new BstVMException("Invalid parameters (line %d)".formatted(ctx.start.getLine())); + } + } + } + + /** + * Pops the top (string) literal, adds a `.' to it if the last non + * '}' character isn't a `.', `?', or `!', and pushes this resulting + * string. + */ + private void bstAddPeriod(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation add.period$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof String s)) { + throw new BstVMException("Can only add a period to a string for add.period$ (line %d)".formatted(ctx.start.getLine())); + } + + Matcher m = ADD_PERIOD_PATTERN.matcher(s); + + if (m.find()) { + StringBuilder sb = new StringBuilder(); + m.appendReplacement(sb, m.group(1)); + sb.append('.'); + String group2 = m.group(2); + if (group2 != null) { + sb.append(m.group(2)); + } + stack.push(sb.toString()); + } else { + stack.push(s); + } + } + + /** + * Executes the function whose name is the entry type of an entry. + * For example if an entry is of type book, this function executes + * the book function. When given as an argument to the ITERATE + * command, call.type$ actually produces the output for the entries. + * For an entry with an unknown type, it executes the function + * default.type. Thus you should define (before the READ command) + * one function for each standard entry type as well as a + * default.type function. + */ + public class BstCallTypeFunction implements BstFunction { + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx) { + throw new BstVMException("Call.type$ can only be called from within a context (ITERATE or REVERSE). (line %d)".formatted(ctx.start.getLine())); + } + + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx, BstEntry bstEntry) { + if (bstEntry == null) { + this.execute(visitor, ctx); // Throw error + } else { + functions.get(bstEntry.entry.getType().getName()).execute(visitor, ctx, bstEntry); + } + } + } + + /** + * Pops the top two (string) literals; it changes the case of the second + * according to the specifications of the first, as follows. (Note: The word + * `letters' in the next sentence refers only to those at brace-level 0, the + * top-most brace level; no other characters are changed, except perhaps for + * \special characters", described in Section 4.) If the first literal is the + * string `t', it converts to lower case all letters except the very first + * character in the string, which it leaves alone, and except the first + * character following any colon and then nonnull white space, which it also + * leaves alone; if it's the string `l', it converts all letters to lower case; + * and if it's the string `u', it converts all letters to upper case. It then + * pushes this resulting string. If either type is incorrect, it complains and + * pushes the null string; however, if both types are correct but the + * specification string (i.e., the first string) isn't one of the legal ones, it + * merely pushes the second back onto the stack, after complaining. (Another + * note: It ignores case differences in the specification string; for example, + * the strings t and T are equivalent for the purposes of this built-in + * function.) + */ + private void bstChangeCase(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation change.case$ (line %d)".formatted(ctx.start.getLine())); + } + + Object o1 = stack.pop(); + if (!((o1 instanceof String format) && (format.length() == 1))) { + throw new BstVMException("A format string of length 1 is needed for change.case$ (line %d)".formatted(ctx.start.getLine())); + } + + Object o2 = stack.pop(); + if (!(o2 instanceof String toChange)) { + throw new BstVMException("A string is needed as second parameter for change.case$ (line %d)".formatted(ctx.start.getLine())); + } + + stack.push(BstCaseChanger.changeCase(toChange, BstCaseChanger.FormatMode.of(format))); + } + + /** + * Pops the top (string) literal, makes sure it's a single + * character, converts it to the corresponding ASCII integer, and + * pushes this integer. + */ + private void bstChrToInt(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation chr.to.int$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!((o1 instanceof String s) && (((String) o1).length() == 1))) { + throw new BstVMException("Can only perform chr.to.int$ on string with length 1 (line %d)".formatted(ctx.start.getLine())); + } + + stack.push((int) s.charAt(0)); + } + + /** + * Pushes the string that was the \cite-command argument for this + * entry. + */ + public class BstCiteFunction implements BstFunction { + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx) { + throw new BstVMException("Must have an entry to cite$ (line %d)".formatted(ctx.start.getLine())); + } + + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx, BstEntry bstEntryContext) { + if (bstEntryContext == null) { + execute(visitor, ctx); + return; + } + + stack.push(bstEntryContext.entry.getCitationKey().orElse(null)); + } + } + + /** + * Pops the top literal from the stack and pushes two copies of it. + */ + private void bstDuplicate(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation duplicate$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + stack.push(o1); + stack.push(o1); + } + + /** + * Pops the top literal and pushes the integer 1 if it's a missing + * field or a string having no non-white-space characters, 0 + * otherwise. + */ + private void bstEmpty(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation empty$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (o1 == null) { + stack.push(BstVM.TRUE); + return; + } + + if (!(o1 instanceof String s)) { + throw new BstVMException("Operand does not match function empty$ (line %d)".formatted(ctx.start.getLine())); + } + + stack.push("".equals(s.trim()) ? BstVM.TRUE : BstVM.FALSE); + } + + /** + * The |built_in| function {\.{format.name\$}} pops the + * top three literals (they are a string, an integer, and a string + * literal, in that order). The last string literal represents a + * name list (each name corresponding to a person), the integer + * literal specifies which name to pick from this list, and the + * first string literal specifies how to format this name, as + * described in the \BibTeX\ documentation. Finally, this function + * pushes the formatted name. If any of the types is incorrect, it + * complains and pushes the null string. + */ + private void bstFormatName(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 3) { + throw new BstVMException("Not enough operands on stack for operation format.name$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + Object o2 = stack.pop(); + Object o3 = stack.pop(); + + if (!(o1 instanceof String) && !(o2 instanceof Integer) && !(o3 instanceof String)) { + // warning("A string is needed for change.case$"); + stack.push(""); + return; + } + + String format = (String) o1; + Integer name = (Integer) o2; + String names = (String) o3; + + if (names == null) { + stack.push(""); + } else { + AuthorList a = AuthorList.parse(names); + if (name > a.getNumberOfAuthors()) { + throw new BstVMException("Author Out of Bounds. Number %d invalid for %s (line %d)".formatted(name, names, ctx.start.getLine())); + } + Author author = a.getAuthor(name - 1); + + stack.push(BstNameFormatter.formatName(author, format)); + } + } + + /** + * Pops the top three literals (they are two function literals and + * an integer literal, in that order); if the integer is greater + * than 0, it executes the second literal, else it executes the + * first. + */ + private void bstIf(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 3) { + throw new BstVMException("Not enough operands on stack for if$ (line %d)".formatted(ctx.start.getLine())); + } + + Object f1 = stack.pop(); + Object f2 = stack.pop(); + Object i = stack.pop(); + + if (!((f1 instanceof BstVMVisitor.Identifier) || (f1 instanceof ParseTree)) + && ((f2 instanceof BstVMVisitor.Identifier) || (f2 instanceof ParseTree)) + && (i instanceof Integer)) { + throw new BstVMException("Expecting two functions and an integer for if$ (line %d)".formatted(ctx.start.getLine())); + } + + if (((Integer) i) > 0) { + callIdentifierOrTree(f2, visitor, ctx); + } else { + callIdentifierOrTree(f1, visitor, ctx); + } + } + + private void callIdentifierOrTree(Object f, BstVMVisitor visitor, ParserRuleContext ctx) { + if (f instanceof ParseTree tree) { + visitor.visit(tree); + } else if (f instanceof BstVMVisitor.Identifier identifier) { + visitor.resolveIdentifier(identifier.name(), ctx); + } else { + stack.push(f); + } + } + + /** + * Pops the top (integer) literal, interpreted as the ASCII integer + * value of a single character, converts it to the corresponding + * single-character string, and pushes this string. + */ + private void bstIntToChr(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation int.to.chr$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof Integer i)) { + throw new BstVMException("Can only perform operation int.to.chr$ on an Integer (line %d)".formatted(ctx.start.getLine())); + } + + stack.push(String.valueOf((char) i.intValue())); + } + + /** + * Pops the top (integer) literal, converts it to its (unique) + * string equivalent, and pushes this string. + */ + private void bstIntToStr(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation int.to.str$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof Integer)) { + throw new BstVMException("Can only transform an integer to an string using int.to.str$ (line %d)".formatted(ctx.start.getLine())); + } + + stack.push(o1.toString()); + } + + /** + * Pops the top literal and pushes the integer 1 if it's a missing + * field, 0 otherwise. + */ + private void bstMissing(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation missing$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (o1 == null) { + stack.push(BstVM.TRUE); + return; + } + + if (!(o1 instanceof String)) { + LOGGER.warn("Not a string or missing field in operation missing$ (line %d)".formatted(ctx.start.getLine())); + stack.push(BstVM.TRUE); + return; + } + + stack.push(BstVM.FALSE); + } + + /** + * Writes onto the bbl file what is accumulated in the output buffer. + * It writes a blank line if and only if the output buffer is empty. + * Since write$ does reasonable line breaking, you should use this + * function only when you want a blank line or an explicit line + * break. + */ + private void bstNewLine(BstVMVisitor visitor, ParserRuleContext ctx) { + this.bbl.append('\n'); + } + + /** + * Pops the top (string) literal and pushes the number of names the + * string represents one plus the number of occurrences of the + * substring "and" (ignoring case differences) surrounded by + * non-null white-space at the top brace level. + */ + private void bstNumNames(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation num.names$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof String s)) { + throw new BstVMException("Need a string at the top of the stack for num.names$ (line %d)".formatted(ctx.start.getLine())); + } + + stack.push(AuthorList.parse(s).getNumberOfAuthors()); + } + + /** + * Pops the top of the stack but doesn't print it; this gets rid of + * an unwanted stack literal. + */ + private void bstPop(BstVMVisitor visitor, ParserRuleContext ctx) { + stack.pop(); + } + + /** + * The |built_in| function {\.{preamble\$}} pushes onto the stack + * the concatenation of all the \.{preamble} strings read from the + * database files. (or the empty string if there were none) + * '@PREAMBLE' strings are read from the database files. + */ + private void bstPreamble(BstVMVisitor visitor, ParserRuleContext ctx) { + stack.push(preamble); + } + + /** + * Pops the top (string) literal, removes nonalphanumeric characters + * except for white-space characters and hyphens and ties (these all get + * converted to a space), removes certain alphabetic characters + * contained in the control sequences associated with a \special + * character", and pushes the resulting string. + */ + private void bstPurify(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation purify$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof String)) { + LOGGER.warn("A string is needed for purify$"); + stack.push(""); + return; + } + + stack.push(BstPurifier.purify((String) o1)); + } + + /** + * Pushes the string consisting of the double-quote character. + */ + private void bstQuote(BstVMVisitor visitor, ParserRuleContext ctx) { + stack.push("\""); + } + + /** + * Does nothing. + */ + private void bstSkip(BstVMVisitor visitor, ParserRuleContext ctx) { + // no-op + } + + /** + * Pops and prints the whole stack; it's meant to be used for style + * designers while debugging. + */ + private void bstStack(BstVMVisitor visitor, ParserRuleContext ctx) { + while (!stack.empty()) { + LOGGER.debug("Stack entry {}", stack.pop()); + } + } + + /** + * Pops the top three literals (they are the two integers literals + * len and start, and a string literal, in that order). It pushes + * the substring of the (at most) len consecutive characters + * starting at the startth character (assuming 1-based indexing) if + * start is positive, and ending at the start-th character + * (including) from the end if start is negative (where the first + * character from the end is the last character). + */ + private void bstSubstring(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 3) { + throw new BstVMException("Not enough operands on stack for operation substring$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + Object o2 = stack.pop(); + Object o3 = stack.pop(); + + if (!((o1 instanceof Integer len) && (o2 instanceof Integer start) && (o3 instanceof String s))) { + throw new BstVMException("Expecting two integers and a string for substring$ (line %d)".formatted(ctx.start.getLine())); + } + + int lenI = len; + int startI = start; + + if (lenI > (Integer.MAX_VALUE / 2)) { + lenI = Integer.MAX_VALUE / 2; + } + + if (startI > (Integer.MAX_VALUE / 2)) { + startI = Integer.MAX_VALUE / 2; + } + + if (startI < (Integer.MIN_VALUE / 2)) { + startI = -Integer.MIN_VALUE / 2; + } + + if (startI < 0) { + startI += s.length() + 1; + startI = Math.max(1, (startI + 1) - lenI); + } + stack.push(s.substring(startI - 1, Math.min((startI - 1) + lenI, s.length()))); + } + + /** + * Swaps the top two literals on the stack. text.length$ Pops the + * top (string) literal, and pushes the number of text characters + * it contains, where an accented character (more precisely, a + * \special character", defined in Section 4) counts as a single + * text character, even if it's missing its matching right brace, + * and where braces don't count as text characters. + */ + private void bstSwap(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation swap$ (line %d)".formatted(ctx.start.getLine())); + } + Object f1 = stack.pop(); + Object f2 = stack.pop(); + + stack.push(f1); + stack.push(f2); + } + + /** + * text.length$ Pops the top (string) literal, and pushes the number + * of text characters it contains, where an accented character (more + * precisely, a "special character", defined in Section 4) counts as + * a single text character, even if it's missing its matching right + * brace, and where braces don't count as text characters. + * + * From BibTeXing: For the purposes of counting letters in labels, + * BibTEX considers everything contained inside the braces as a + * single letter. + */ + private void bstTextLength(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation text.length$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof String s)) { + throw new BstVMException("Can only perform operation on a string text.length$ (line %d)".formatted(ctx.start.getLine())); + } + + char[] c = s.toCharArray(); + int result = 0; + int i = 0; + int n = s.length(); + int braceLevel = 0; + + while (i < n) { + i++; + if (c[i - 1] == '{') { + braceLevel++; + if ((braceLevel == 1) && (i < n)) { + if (c[i] == '\\') { + i++; // skip over backslash + while ((i < n) && (braceLevel > 0)) { + if (c[i] == '}') { + braceLevel--; + } else if (c[i] == '{') { + braceLevel++; + } + i++; + } + result++; + } + } + } else if (c[i - 1] == '}') { + if (braceLevel > 0) { + braceLevel--; + } + } else { + result++; + } + } + stack.push(result); + } + + /** + * Pops the top two literals (the integer literal len and a string + * literal, in that order). It pushes the substring of the (at most) len + * consecutive text characters starting from the beginning of the + * string. This function is similar to substring$, but this one + * considers a \special character", even if it's missing its matching + * right brace, to be a single text character (rather than however many + * ASCII characters it actually comprises), and this function doesn't + * consider braces to be text characters; furthermore, this function + * appends any needed matching right braces. + */ + private void bstTextPrefix(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation text.prefix$ (line %d)".formatted(ctx.start.getLine())); + } + + Object o1 = stack.pop(); + if (!(o1 instanceof Integer)) { + LOGGER.warn("An integer is needed as first parameter to text.prefix$ (line {})", ctx.start.getLine()); + stack.push(""); + return; + } + + Object o2 = stack.pop(); + if (!(o2 instanceof String)) { + LOGGER.warn("A string is needed as second parameter to text.prefix$ (line {})", ctx.start.getLine()); + stack.push(""); + return; + } + + stack.push(BstTextPrefixer.textPrefix((Integer) o1, (String) o2)); + } + + /** + * Pops and prints the top of the stack to the log file. It's useful for debugging. + */ + private void bstTop(BstVMVisitor visitor, ParserRuleContext ctx) { + LOGGER.debug("Stack entry {} (line {})", stack.pop(), ctx.start.getLine()); + } + + /** + * Pushes the current entry's type (book, article, etc.), but pushes + * the null string if the type is either unknown or undefined. + */ + public class BstTypeFunction implements BstFunction { + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx) { + throw new BstVMException("type$ need a context (line %d)".formatted(ctx.start.getLine())); + } + + @Override + public void execute(BstVMVisitor visitor, ParserRuleContext ctx, BstEntry bstEntryContext) { + if (bstEntryContext == null) { + this.execute(visitor, ctx); + return; + } + + stack.push(bstEntryContext.entry.getType().getName()); + } + } + + /** + * Pops the top (string) literal and prints it following a warning + * message. This also increments a count of the number of warning + * messages issued. + */ + private void bstWarning(BstVMVisitor visitor, ParserRuleContext ctx) { + LOGGER.warn("Warning (#{}): {}", bstWarning++, stack.pop()); + } + + /** + * Pops the top two (function) literals, and keeps executing the + * second as long as the (integer) literal left on the stack by + * executing the first is greater than 0. + */ + private void bstWhile(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.size() < 2) { + throw new BstVMException("Not enough operands on stack for operation while$ (line %d)".formatted(ctx.start.getLine())); + } + Object f2 = stack.pop(); + Object f1 = stack.pop(); + + if (!((f1 instanceof BstVMVisitor.Identifier) || (f1 instanceof ParseTree)) + && ((f2 instanceof BstVMVisitor.Identifier) || (f2 instanceof ParseTree))) { + throw new BstVMException("Expecting two functions for while$ (line %d)".formatted(ctx.start.getLine())); + } + + do { + visitor.visit((ParseTree) f1); + + Object i = stack.pop(); + if (!(i instanceof Integer)) { + throw new BstVMException("First parameter to while has to return an integer but was %s (line %d)" + .formatted(i.toString(), ctx.start.getLine())); + } + if ((Integer) i <= 0) { + break; + } + visitor.visit((ParseTree) f2); + } while (true); + } + + /** + * The |built_in| function {\.{width\$}} pops the top (string) literal and + * pushes the integer that represents its width in units specified by the + * |char_width| array. This function takes the literal literally; that is, it + * assumes each character in the string is to be printed as is, regardless of + * whether the character has a special meaning to \TeX, except that special + * characters (even without their |right_brace|s) are handled specially. If the + * literal isn't a string, it complains and pushes~0. + */ + private void bstWidth(BstVMVisitor visitor, ParserRuleContext ctx) { + if (stack.isEmpty()) { + throw new BstVMException("Not enough operands on stack for operation width$ (line %d)".formatted(ctx.start.getLine())); + } + Object o1 = stack.pop(); + + if (!(o1 instanceof String)) { + LOGGER.warn("A string is needed for width$"); + stack.push(0); + return; + } + + stack.push(BstWidthCalculator.width((String) o1)); + } + + /** + * Pops the top (string) literal and writes it on the output buffer + * (which will result in stuff being written onto the bbl file when + * the buffer fills up). + */ + private void bstWrite(BstVMVisitor visitor, ParserRuleContext ctx) { + String s = (String) stack.pop(); + bbl.append(s); + } +} diff --git a/src/main/java/org/jabref/logic/bst/BstPreviewLayout.java b/src/main/java/org/jabref/logic/bst/BstPreviewLayout.java index d0c040842fd..a002d388e60 100644 --- a/src/main/java/org/jabref/logic/bst/BstPreviewLayout.java +++ b/src/main/java/org/jabref/logic/bst/BstPreviewLayout.java @@ -23,7 +23,7 @@ public class BstPreviewLayout implements PreviewLayout { private final String name; - private VM vm; + private BstVM bstVM; private String error; public BstPreviewLayout(Path path) { @@ -34,7 +34,7 @@ public BstPreviewLayout(Path path) { return; } try { - vm = new VM(path.toFile()); + bstVM = new BstVM(path); } catch (Exception e) { LOGGER.error("Could not read {}.", path.toAbsolutePath(), e); error = Localization.lang("Error opening file '%0'.", path.toString()); @@ -49,7 +49,7 @@ public String generatePreview(BibEntry originalEntry, BibDatabaseContext databas // ensure that the entry is of BibTeX format (and do not modify the original entry) BibEntry entry = (BibEntry) originalEntry.clone(); new ConvertToBibtexCleanup().cleanup(entry); - String result = vm.run(List.of(entry)); + String result = bstVM.render(List.of(entry)); // Remove all comments result = result.replaceAll("%.*", ""); // Remove all LaTeX comments diff --git a/src/main/java/org/jabref/logic/bst/BstVM.java b/src/main/java/org/jabref/logic/bst/BstVM.java new file mode 100644 index 00000000000..a2428d683d9 --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/BstVM.java @@ -0,0 +1,114 @@ +package org.jabref.logic.bst; + +import java.io.IOException; +import java.nio.file.Path; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.Objects; +import java.util.Stack; + +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; + +import org.antlr.v4.runtime.BailErrorStrategy; +import org.antlr.v4.runtime.BaseErrorListener; +import org.antlr.v4.runtime.CharStream; +import org.antlr.v4.runtime.CharStreams; +import org.antlr.v4.runtime.CommonTokenStream; +import org.antlr.v4.runtime.RecognitionException; +import org.antlr.v4.runtime.Recognizer; +import org.antlr.v4.runtime.misc.ParseCancellationException; +import org.antlr.v4.runtime.tree.ParseTree; + +public class BstVM { + + protected static final Integer FALSE = 0; + protected static final Integer TRUE = 1; + + protected final ParseTree tree; + protected BstVMContext latestContext; // for testing + + private Path path = null; + + public BstVM(Path path) throws RecognitionException, IOException { + this(CharStreams.fromPath(path)); + this.path = path; + } + + public BstVM(String s) throws RecognitionException { + this(CharStreams.fromString(s)); + } + + protected BstVM(CharStream bst) throws RecognitionException { + this(charStream2CommonTree(bst)); + } + + private BstVM(ParseTree tree) { + this.tree = tree; + } + + private static ParseTree charStream2CommonTree(CharStream query) { + BstLexer lexer = new BstLexer(query); + lexer.removeErrorListeners(); + lexer.addErrorListener(ThrowingErrorListener.INSTANCE); + BstParser parser = new BstParser(new CommonTokenStream(lexer)); + parser.removeErrorListeners(); + parser.addErrorListener(ThrowingErrorListener.INSTANCE); + parser.setErrorHandler(new BailErrorStrategy()); + return parser.bstFile(); + } + + /** + * Transforms the given list of BibEntries to a rendered list of references using the parsed bst program + * + * @param bibEntries list of entries to convert + * @param bibDatabase (may be null) the bibDatabase used for resolving strings / crossref + * @return list of references in plain text form + */ + public String render(Collection bibEntries, BibDatabase bibDatabase) { + Objects.requireNonNull(bibEntries); + + List entries = new ArrayList<>(bibEntries.size()); + for (BibEntry entry : bibEntries) { + entries.add(new BstEntry(entry)); + } + + StringBuilder resultBuffer = new StringBuilder(); + + BstVMContext bstVMContext = new BstVMContext(entries, bibDatabase, path); + bstVMContext.functions().putAll(new BstFunctions(bstVMContext, resultBuffer).getBuiltInFunctions()); + bstVMContext.integers().put("entry.max$", Integer.MAX_VALUE); + bstVMContext.integers().put("global.max$", Integer.MAX_VALUE); + + BstVMVisitor bstVMVisitor = new BstVMVisitor(bstVMContext, resultBuffer); + bstVMVisitor.visit(tree); + + latestContext = bstVMContext; + + return resultBuffer.toString(); + } + + public String render(Collection bibEntries) { + return render(bibEntries, null); + } + + protected Stack getStack() { + if (latestContext != null) { + return latestContext.stack(); + } else { + throw new BstVMException("BstVM must have rendered at least once to provide the latest stack"); + } + } + + private static class ThrowingErrorListener extends BaseErrorListener { + public static final ThrowingErrorListener INSTANCE = new ThrowingErrorListener(); + + @Override + public void syntaxError(Recognizer, ?> recognizer, Object offendingSymbol, + int line, int charPositionInLine, String msg, RecognitionException e) + throws ParseCancellationException { + throw new ParseCancellationException("line " + line + ":" + charPositionInLine + " " + msg); + } + } +} diff --git a/src/main/java/org/jabref/logic/bst/BstVMContext.java b/src/main/java/org/jabref/logic/bst/BstVMContext.java new file mode 100644 index 00000000000..8a98f605dd8 --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/BstVMContext.java @@ -0,0 +1,22 @@ +package org.jabref.logic.bst; + +import java.nio.file.Path; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.Stack; + +import org.jabref.model.database.BibDatabase; + +public record BstVMContext(List entries, + Map strings, + Map integers, + Map functions, + Stack stack, + BibDatabase bibDatabase, + Optional path) { + public BstVMContext(List entries, BibDatabase bibDatabase, Path path) { + this(entries, new HashMap<>(), new HashMap<>(), new HashMap<>(), new Stack<>(), bibDatabase, Optional.ofNullable(path)); + } +} diff --git a/src/main/java/org/jabref/logic/bst/BstVMException.java b/src/main/java/org/jabref/logic/bst/BstVMException.java new file mode 100644 index 00000000000..2851166d094 --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/BstVMException.java @@ -0,0 +1,7 @@ +package org.jabref.logic.bst; + +public class BstVMException extends RuntimeException { + public BstVMException(String string) { + super(string); + } +} diff --git a/src/main/java/org/jabref/logic/bst/BstVMVisitor.java b/src/main/java/org/jabref/logic/bst/BstVMVisitor.java new file mode 100644 index 00000000000..a815480e506 --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/BstVMVisitor.java @@ -0,0 +1,263 @@ +package org.jabref.logic.bst; + +import java.util.Comparator; +import java.util.List; +import java.util.ListIterator; +import java.util.Map; + +import org.jabref.logic.bibtex.FieldContentFormatterPreferences; +import org.jabref.logic.bibtex.FieldWriter; +import org.jabref.logic.bibtex.FieldWriterPreferences; +import org.jabref.logic.bibtex.InvalidFieldValueException; +import org.jabref.model.entry.Month; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.field.StandardField; + +import org.antlr.v4.runtime.ParserRuleContext; +import org.antlr.v4.runtime.tree.ParseTree; +import org.antlr.v4.runtime.tree.TerminalNode; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +class BstVMVisitor extends BstBaseVisitor { + private static final Logger LOGGER = LoggerFactory.getLogger(BstVMVisitor.class); + + private final BstVMContext bstVMContext; + private final StringBuilder bbl; + + private BstEntry selectedBstEntry = null; + + public record Identifier(String name) { + } + + public BstVMVisitor(BstVMContext bstVMContext, StringBuilder bbl) { + this.bstVMContext = bstVMContext; + this.bbl = bbl; + } + + @Override + public Integer visitStringsCommand(BstParser.StringsCommandContext ctx) { + if (ctx.ids.identifier().size() > 20) { + throw new BstVMException("Strings limit reached"); + } + + for (BstParser.IdentifierContext identifierContext : ctx.ids.identifier()) { + bstVMContext.strings().put(identifierContext.getText(), null); + } + return BstVM.TRUE; + } + + @Override + public Integer visitIntegersCommand(BstParser.IntegersCommandContext ctx) { + for (BstParser.IdentifierContext identifierContext : ctx.ids.identifier()) { + bstVMContext.integers().put(identifierContext.getText(), 0); + } + return BstVM.TRUE; + } + + @Override + public Integer visitFunctionCommand(BstParser.FunctionCommandContext ctx) { + bstVMContext.functions().put(ctx.id.getText(), + (visitor, functionContext) -> visitor.visit(ctx.function)); + return BstVM.TRUE; + } + + @Override + public Integer visitMacroCommand(BstParser.MacroCommandContext ctx) { + String replacement = ctx.repl.getText().substring(1, ctx.repl.getText().length() - 1); + bstVMContext.functions().put(ctx.id.getText(), + (visitor, functionContext) -> bstVMContext.stack().push(replacement)); + return BstVM.TRUE; + } + + @Override + public Integer visitReadCommand(BstParser.ReadCommandContext ctx) { + FieldWriter fieldWriter = new FieldWriter(new FieldWriterPreferences(true, List.of(StandardField.MONTH), new FieldContentFormatterPreferences())); + for (BstEntry e : bstVMContext.entries()) { + for (Map.Entry mEntry : e.fields.entrySet()) { + Field field = FieldFactory.parseField(mEntry.getKey()); + String fieldValue = e.entry.getResolvedFieldOrAlias(field, bstVMContext.bibDatabase()) + .map(content -> { + try { + String result = fieldWriter.write(field, content); + if (result.startsWith("{")) { + // Strip enclosing {} from the output + return result.substring(1, result.length() - 1); + } + if (field == StandardField.MONTH) { + // We don't have the internal BibTeX strings at hand. + // Thus, we look up the full month name in the generic table. + return Month.parse(result) + .map(Month::getFullName) + .orElse(result); + } + return result; + } catch ( + InvalidFieldValueException invalidFieldValueException) { + // in case there is something wrong with the content, just return the content itself + return content; + } + }) + .orElse(null); + mEntry.setValue(fieldValue); + } + } + + for (BstEntry e : bstVMContext.entries()) { + if (!e.fields.containsKey(StandardField.CROSSREF.getName())) { + e.fields.put(StandardField.CROSSREF.getName(), null); + } + } + + return BstVM.TRUE; + } + + @Override + public Integer visitExecuteCommand(BstParser.ExecuteCommandContext ctx) { + this.selectedBstEntry = null; + visit(ctx.bstFunction()); + + return BstVM.TRUE; + } + + @Override + public Integer visitIterateCommand(BstParser.IterateCommandContext ctx) { + for (BstEntry entry : bstVMContext.entries()) { + this.selectedBstEntry = entry; + visit(ctx.bstFunction()); + } + + return BstVM.TRUE; + } + + @Override + public Integer visitReverseCommand(BstParser.ReverseCommandContext ctx) { + ListIterator i = bstVMContext.entries().listIterator(bstVMContext.entries().size()); + while (i.hasPrevious()) { + this.selectedBstEntry = i.previous(); + visit(ctx.bstFunction()); + } + + return BstVM.TRUE; + } + + @Override + public Integer visitEntryCommand(BstParser.EntryCommandContext ctx) { + // ENTRY command contains 3 optionally filled identifier lists: + // Fields, Integers and Strings + + BstParser.IdListOptContext entryFields = ctx.idListOpt(0); + for (BstParser.IdentifierContext identifierContext : entryFields.identifier()) { + for (BstEntry entry : bstVMContext.entries()) { + entry.fields.put(identifierContext.getText(), null); + } + } + + BstParser.IdListOptContext entryIntegers = ctx.idListOpt(1); + for (BstParser.IdentifierContext identifierContext : entryIntegers.identifier()) { + for (BstEntry entry : bstVMContext.entries()) { + entry.localIntegers.put(identifierContext.getText(), 0); + } + } + + BstParser.IdListOptContext entryStrings = ctx.idListOpt(2); + for (BstParser.IdentifierContext identifierContext : entryStrings.identifier()) { + for (BstEntry entry : bstVMContext.entries()) { + entry.localStrings.put(identifierContext.getText(), null); + } + } + + for (BstEntry entry : bstVMContext.entries()) { + entry.localStrings.put("sort.key$", null); + } + + return BstVM.TRUE; + } + + @Override + public Integer visitSortCommand(BstParser.SortCommandContext ctx) { + bstVMContext.entries().sort(Comparator.comparing(o -> (o.localStrings.get("sort.key$")))); + return BstVM.TRUE; + } + + @Override + public Integer visitIdentifier(BstParser.IdentifierContext ctx) { + resolveIdentifier(ctx.IDENTIFIER().getText(), ctx); + return BstVM.TRUE; + } + + protected void resolveIdentifier(String name, ParserRuleContext ctx) { + if (selectedBstEntry != null) { + if (selectedBstEntry.fields.containsKey(name)) { + bstVMContext.stack().push(selectedBstEntry.fields.get(name)); + return; + } + if (selectedBstEntry.localStrings.containsKey(name)) { + bstVMContext.stack().push(selectedBstEntry.localStrings.get(name)); + return; + } + if (selectedBstEntry.localIntegers.containsKey(name)) { + bstVMContext.stack().push(selectedBstEntry.localIntegers.get(name)); + return; + } + } + + if (bstVMContext.strings().containsKey(name)) { + bstVMContext.stack().push(bstVMContext.strings().get(name)); + return; + } + if (bstVMContext.integers().containsKey(name)) { + bstVMContext.stack().push(bstVMContext.integers().get(name)); + return; + } + if (bstVMContext.functions().containsKey(name)) { + bstVMContext.functions().get(name).execute(this, ctx); + return; + } + + throw new BstVMException("No matching identifier found: " + name); + } + + @Override + public Integer visitBstFunction(BstParser.BstFunctionContext ctx) { + String name = ctx.getChild(0).getText(); + if (bstVMContext.functions().containsKey(name)) { + bstVMContext.functions().get(name).execute(this, ctx, selectedBstEntry); + } else { + visit(ctx.getChild(0)); + } + + return BstVM.TRUE; + } + + @Override + public Integer visitStackitem(BstParser.StackitemContext ctx) { + for (ParseTree childNode : ctx.children) { + try { + if (childNode instanceof TerminalNode token) { + switch (token.getSymbol().getType()) { + case BstParser.STRING -> { + String s = token.getText(); + bstVMContext.stack().push(s.substring(1, s.length() - 1)); + } + case BstParser.INTEGER -> + bstVMContext.stack().push(Integer.parseInt(token.getText().substring(1))); + case BstParser.QUOTED -> + bstVMContext.stack().push(new Identifier(token.getText().substring(1))); + } + } else if (childNode instanceof BstParser.StackContext) { + bstVMContext.stack().push(childNode); + } else { + this.visit(childNode); + } + } catch (BstVMException e) { + bstVMContext.path().ifPresentOrElse( + (path) -> LOGGER.error("{} ({})", e.getMessage(), path), + () -> LOGGER.error(e.getMessage())); + throw e; + } + } + return BstVM.TRUE; + } +} diff --git a/src/main/java/org/jabref/logic/bst/ChangeCaseFunction.java b/src/main/java/org/jabref/logic/bst/ChangeCaseFunction.java deleted file mode 100644 index 65f4f64c619..00000000000 --- a/src/main/java/org/jabref/logic/bst/ChangeCaseFunction.java +++ /dev/null @@ -1,65 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Locale; -import java.util.Stack; - -import org.jabref.logic.bst.BibtexCaseChanger.FORMAT_MODE; -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; - -/** - * From the Bibtex manual: - * - * Pops the top two (string) literals; it changes the case of the second - * according to the specifications of the first, as follows. (Note: The word - * `letters' in the next sentence refers only to those at brace-level 0, the - * top-most brace level; no other characters are changed, except perhaps for - * \special characters", described in Section 4.) If the first literal is the - * string `t', it converts to lower case all letters except the very first - * character in the string, which it leaves alone, and except the first - * character following any colon and then nonnull white space, which it also - * leaves alone; if it's the string `l', it converts all letters to lower case; - * and if it's the string `u', it converts all letters to upper case. It then - * pushes this resulting string. If either type is incorrect, it complains and - * pushes the null string; however, if both types are correct but the - * specification string (i.e., the first string) isn't one of the legal ones, it - * merely pushes the second back onto the stack, after complaining. (Another - * note: It ignores case differences in the specification string; for example, - * the strings t and T are equivalent for the purposes of this built-in - * function.) - * - * Christopher: I think this should be another grammar! This parser is horrible. - * - */ -public class ChangeCaseFunction implements BstFunction { - - private final VM vm; - - public ChangeCaseFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation change.case$"); - } - - Object o1 = stack.pop(); - if (!((o1 instanceof String) && (((String) o1).length() == 1))) { - throw new VMException("A format string of length 1 is needed for change.case$"); - } - - Object o2 = stack.pop(); - if (!(o2 instanceof String)) { - throw new VMException("A string is needed as second parameter for change.case$"); - } - - char format = ((String) o1).toLowerCase(Locale.ROOT).charAt(0); - String s = (String) o2; - - stack.push(BibtexCaseChanger.changeCase(s, FORMAT_MODE.getFormatModeForBSTFormat(format))); - } -} diff --git a/src/main/java/org/jabref/logic/bst/FormatNameFunction.java b/src/main/java/org/jabref/logic/bst/FormatNameFunction.java deleted file mode 100644 index b2845c89be7..00000000000 --- a/src/main/java/org/jabref/logic/bst/FormatNameFunction.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Stack; - -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; -import org.jabref.model.entry.Author; -import org.jabref.model.entry.AuthorList; - -/** - * From Bibtex: - * - * "The |built_in| function {\.{format.name\$}} pops the - * top three literals (they are a string, an integer, and a string - * literal, in that order). The last string literal represents a - * name list (each name corresponding to a person), the integer - * literal specifies which name to pick from this list, and the - * first string literal specifies how to format this name, as - * described in the \BibTeX\ documentation. Finally, this function - * pushes the formatted name. If any of the types is incorrect, it - * complains and pushes the null string." - * - * All the pain is encapsulated in BibtexNameFormatter. :-) - * - */ -public class FormatNameFunction implements BstFunction { - - private final VM vm; - - public FormatNameFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.size() < 3) { - throw new VMException("Not enough operands on stack for operation format.name$"); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - Object o3 = stack.pop(); - - if (!(o1 instanceof String) && !(o2 instanceof Integer) && !(o3 instanceof String)) { - // warning("A string is needed for change.case$"); - stack.push(""); - return; - } - - String format = (String) o1; - Integer name = (Integer) o2; - String names = (String) o3; - - if (names == null) { - stack.push(""); - } else { - AuthorList a = AuthorList.parse(names); - if (name > a.getNumberOfAuthors()) { - throw new VMException("Author Out of Bounds. Number " + name + " invalid for " + names); - } - Author author = a.getAuthor(name - 1); - - stack.push(BibtexNameFormatter.formatName(author, format, vm)); - } - } -} diff --git a/src/main/java/org/jabref/logic/bst/PurifyFunction.java b/src/main/java/org/jabref/logic/bst/PurifyFunction.java deleted file mode 100644 index b0df9df3735..00000000000 --- a/src/main/java/org/jabref/logic/bst/PurifyFunction.java +++ /dev/null @@ -1,43 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Stack; - -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; - -/** - * - * The |built_in| function {\.{purify\$}} pops the top (string) literal, removes - * nonalphanumeric characters except for |white_space| and |sep_char| characters - * (these get converted to a |space|) and removes certain alphabetic characters - * contained in the control sequences associated with a special character, and - * pushes the resulting string. If the literal isn't a string, it complains and - * pushes the null string. - * - */ -public class PurifyFunction implements BstFunction { - - private final VM vm; - - public PurifyFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation purify$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - vm.warn("A string is needed for purify$"); - stack.push(""); - return; - } - - stack.push(BibtexPurify.purify((String) o1, vm)); - } -} diff --git a/src/main/java/org/jabref/logic/bst/TextPrefixFunction.java b/src/main/java/org/jabref/logic/bst/TextPrefixFunction.java deleted file mode 100644 index 415ee97498b..00000000000 --- a/src/main/java/org/jabref/logic/bst/TextPrefixFunction.java +++ /dev/null @@ -1,54 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Stack; - -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; - -/** - * The |built_in| function {\.{text.prefix\$}} pops the top two literals - * (the integer literal |pop_lit1| and a string literal, in that order). - * It pushes the substring of the (at most) |pop_lit1| consecutive text - * characters starting from the beginning of the string. This function - * is similar to {\.{substring\$}}, but this one considers an accented - * character (or more precisely, a ``special character''$\!$, even if - * it's missing its matching |right_brace|) to be a single text character - * (rather than however many |ASCII_code| characters it actually - * comprises), and this function doesn't consider braces to be text - * characters; furthermore, this function appends any needed matching - * |right_brace|s. If any of the types is incorrect, it complains and - * pushes the null string. - */ -public class TextPrefixFunction implements BstFunction { - - private final VM vm; - - public TextPrefixFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation text.prefix$"); - } - - Object o1 = stack.pop(); - if (!(o1 instanceof Integer)) { - vm.warn("An integer is needed as first parameter to text.prefix$"); - stack.push(""); - return; - } - - Object o2 = stack.pop(); - if (!(o2 instanceof String)) { - vm.warn("A string is needed as second parameter to text.prefix$"); - stack.push(""); - return; - } - - stack.push(BibtexTextPrefix.textPrefix((Integer) o1, (String) o2, vm)); - } -} diff --git a/src/main/java/org/jabref/logic/bst/VM.java b/src/main/java/org/jabref/logic/bst/VM.java deleted file mode 100644 index caaf9078f64..00000000000 --- a/src/main/java/org/jabref/logic/bst/VM.java +++ /dev/null @@ -1,1245 +0,0 @@ -package org.jabref.logic.bst; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.Collection; -import java.util.Comparator; -import java.util.HashMap; -import java.util.List; -import java.util.ListIterator; -import java.util.Map; -import java.util.Objects; -import java.util.Stack; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -import org.jabref.logic.bibtex.FieldContentFormatterPreferences; -import org.jabref.logic.bibtex.FieldWriter; -import org.jabref.logic.bibtex.FieldWriterPreferences; -import org.jabref.logic.bibtex.InvalidFieldValueException; -import org.jabref.model.database.BibDatabase; -import org.jabref.model.entry.AuthorList; -import org.jabref.model.entry.BibEntry; -import org.jabref.model.entry.Month; -import org.jabref.model.entry.field.Field; -import org.jabref.model.entry.field.FieldFactory; -import org.jabref.model.entry.field.StandardField; - -import org.antlr.runtime.ANTLRFileStream; -import org.antlr.runtime.ANTLRStringStream; -import org.antlr.runtime.CharStream; -import org.antlr.runtime.CommonTokenStream; -import org.antlr.runtime.RecognitionException; -import org.antlr.runtime.tree.CommonTree; -import org.antlr.runtime.tree.Tree; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -/** - * A BibTeX Virtual machine that can execute .bst files. - * - * Documentation can be found in the original bibtex distribution: - * - * https://www.ctan.org/pkg/bibtex - */ -public class VM implements Warn { - - public static final Integer FALSE = 0; - - public static final Integer TRUE = 1; - - private static final Pattern ADD_PERIOD_PATTERN = Pattern.compile("([^\\.\\?\\!\\}\\s])(\\}|\\s)*$"); - - private static final Logger LOGGER = LoggerFactory.getLogger(VM.class); - - private List entries; - - private Map strings = new HashMap<>(); - - private Map integers = new HashMap<>(); - - private Map functions = new HashMap<>(); - - private Stack stack = new Stack<>(); - - private final Map buildInFunctions; - - private File file; - - private final CommonTree tree; - - private StringBuilder bbl; - - private String preamble = ""; - - public static class Identifier { - - public final String name; - - public Identifier(String name) { - this.name = name; - } - - public String getName() { - return name; - } - } - - public static class Variable { - - public final String name; - - public Variable(String name) { - this.name = name; - } - - public String getName() { - return name; - } - } - - @FunctionalInterface - public interface BstFunction { - void execute(BstEntry context); - } - - public VM(File f) throws RecognitionException, IOException { - this(new ANTLRFileStream(f.getPath())); - this.file = f; - } - - public VM(String s) throws RecognitionException { - this(new ANTLRStringStream(s)); - } - - private VM(CharStream bst) throws RecognitionException { - this(VM.charStream2CommonTree(bst)); - } - - private VM(CommonTree tree) { - this.tree = tree; - - this.buildInFunctions = new HashMap<>(37); - - /* - * Pops the top two (integer) literals, compares them, and pushes - * the integer 1 if the second is greater than the first, 0 - * otherwise. - */ - buildInFunctions.put(">", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation >"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only compare two integers with >"); - } - - stack.push(((Integer) o1).compareTo((Integer) o2) > 0 ? VM.TRUE : VM.FALSE); - }); - - /* Analogous to >. */ - buildInFunctions.put("<", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation <"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only compare two integers with <"); - } - - stack.push(((Integer) o1).compareTo((Integer) o2) < 0 ? VM.TRUE : VM.FALSE); - }); - - /* - * Pops the top two (both integer or both string) literals, compares - * them, and pushes the integer 1 if they're equal, 0 otherwise. - */ - buildInFunctions.put("=", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation ="); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - - if ((o1 == null) ^ (o2 == null)) { - stack.push(VM.FALSE); - return; - } - - if ((o1 == null) && (o2 == null)) { - stack.push(VM.TRUE); - return; - } - - stack.push(o1.equals(o2) ? VM.TRUE : VM.FALSE); - }); - - /* Pops the top two (integer) literals and pushes their sum. */ - buildInFunctions.put("+", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation +"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only compare two integers with +"); - } - - stack.push((Integer) o1 + (Integer) o2); - }); - - /* - * Pops the top two (integer) literals and pushes their difference - * (the first subtracted from the second). - */ - buildInFunctions.put("-", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation -"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only subtract two integers with -"); - } - - stack.push((Integer) o1 - (Integer) o2); - }); - - /* - * Pops the top two (string) literals, concatenates them (in reverse - * order, that is, the order in which pushed), and pushes the - * resulting string. - */ - buildInFunctions.put("*", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation *"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (o1 == null) { - o1 = ""; - } - if (o2 == null) { - o2 = ""; - } - - if (!((o1 instanceof String) && (o2 instanceof String))) { - LOGGER.error("o1: {} ({})", o1, o1.getClass()); - LOGGER.error("o2: {} ({})", o2, o2.getClass()); - throw new VMException("Can only concatenate two String with *"); - } - - stack.push(o1.toString() + o2); - }); - - /* - * Pops the top two literals and assigns to the first (which must be - * a global or entry variable) the value of the second. - */ - buildInFunctions.put(":=", context -> { - if (stack.size() < 2) { - throw new VMException("Invalid call to operation :="); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - assign(context, o1, o2); - }); - - /* - * Pops the top (string) literal, adds a `.' to it if the last non - * '}' character isn't a `.', `?', or `!', and pushes this resulting - * string. - */ - buildInFunctions.put("add.period$", context -> addPeriodFunction()); - - /* - * Executes the function whose name is the entry type of an entry. - * For example if an entry is of type book, this function executes - * the book function. When given as an argument to the ITERATE - * command, call.type$ actually produces the output for the entries. - * For an entry with an unknown type, it executes the function - * default.type. Thus you should define (before the READ command) - * one function for each standard entry type as well as a - * default.type function. - */ - buildInFunctions.put("call.type$", context -> { - if (context == null) { - throw new VMException("Call.type$ can only be called from within a context (ITERATE or REVERSE)."); - } - VM.this.execute(context.entry.getType().getName(), context); - }); - - buildInFunctions.put("change.case$", new ChangeCaseFunction(this)); - - /* - * Pops the top (string) literal, makes sure it's a single - * character, converts it to the corresponding ASCII integer, and - * pushes this integer. - */ - buildInFunctions.put("chr.to.int$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation chr.to.int$"); - } - Object o1 = stack.pop(); - - if (!((o1 instanceof String) && (((String) o1).length() == 1))) { - throw new VMException("Can only perform chr.to.int$ on string with length 1"); - } - - String s = (String) o1; - - stack.push((int) s.charAt(0)); - }); - - /* - * Pushes the string that was the \cite-command argument for this - * entry. - */ - buildInFunctions.put("cite$", context -> { - if (context == null) { - throw new VMException("Must have an entry to cite$"); - } - stack.push(context.entry.getCitationKey().orElse(null)); - }); - - /* - * Pops the top literal from the stack and pushes two copies of it. - */ - buildInFunctions.put("duplicate$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation duplicate$"); - } - Object o1 = stack.pop(); - - stack.push(o1); - stack.push(o1); - }); - - /* - * Pops the top literal and pushes the integer 1 if it's a missing - * field or a string having no non-white-space characters, 0 - * otherwise. - */ - buildInFunctions.put("empty$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation empty$"); - } - Object o1 = stack.pop(); - - if (o1 == null) { - stack.push(VM.TRUE); - return; - } - - if (!(o1 instanceof String)) { - throw new VMException("Operand does not match function empty$"); - } - - String s = (String) o1; - - stack.push("".equals(s.trim()) ? VM.TRUE : VM.FALSE); - }); - - buildInFunctions.put("format.name$", new FormatNameFunction(this)); - - /* - * Pops the top three literals (they are two function literals and - * an integer literal, in that order); if the integer is greater - * than 0, it executes the second literal, else it executes the - * first. - */ - buildInFunctions.put("if$", context -> { - if (stack.size() < 3) { - throw new VMException("Not enough operands on stack for operation ="); - } - Object f1 = stack.pop(); - Object f2 = stack.pop(); - Object i = stack.pop(); - - if (!((f1 instanceof Identifier) || (f1 instanceof Tree)) - && ((f2 instanceof Identifier) || (f2 instanceof Tree)) && (i instanceof Integer)) { - throw new VMException("Expecting two functions and an integer for if$."); - } - - if ((Integer) i > 0) { - VM.this.executeInContext(f2, context); - } else { - VM.this.executeInContext(f1, context); - } - }); - - /* - * Pops the top (integer) literal, interpreted as the ASCII integer - * value of a single character, converts it to the corresponding - * single-character string, and pushes this string. - */ - buildInFunctions.put("int.to.chr$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation int.to.chr$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof Integer)) { - throw new VMException("Can only perform operation int.to.chr$ on an Integer"); - } - - Integer i = (Integer) o1; - - stack.push(String.valueOf((char) i.intValue())); - }); - - /* - * Pops the top (integer) literal, converts it to its (unique) - * string equivalent, and pushes this string. - */ - buildInFunctions.put("int.to.str$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation int.to.str$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof Integer)) { - throw new VMException("Can only transform an integer to an string using int.to.str$"); - } - - stack.push(o1.toString()); - }); - - /* - * Pops the top literal and pushes the integer 1 if it's a missing - * field, 0 otherwise. - */ - buildInFunctions.put("missing$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation missing$"); - } - Object o1 = stack.pop(); - - if (o1 == null) { - stack.push(VM.TRUE); - return; - } - - if (!(o1 instanceof String)) { - warn("Not a string or missing field in operation missing$"); - stack.push(VM.TRUE); - return; - } - - stack.push(VM.FALSE); - }); - - /* - * Writes onto the bbl file what is accumulated in the output buffer. - * It writes a blank line if and only if the output buffer is empty. - * Since write$ does reasonable line breaking, you should use this - * function only when you want a blank line or an explicit line - * break. - */ - buildInFunctions.put("newline$", context -> VM.this.bbl.append('\n')); - - /* - * Pops the top (string) literal and pushes the number of names the - * string represents one plus the number of occurrences of the - * substring "and" (ignoring case differences) surrounded by - * non-null white-space at the top brace level. - */ - buildInFunctions.put("num.names$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation num.names$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - throw new VMException("Need a string at the top of the stack for num.names$"); - } - String s = (String) o1; - - stack.push(AuthorList.parse(s).getNumberOfAuthors()); - }); - - /* - * Pops the top of the stack but doesn't print it; this gets rid of - * an unwanted stack literal. - */ - buildInFunctions.put("pop$", context -> stack.pop()); - - /* - * The |built_in| function {\.{preamble\$}} pushes onto the stack - * the concatenation of all the \.{preamble} strings read from the - * database files. (or the empty string if there where none) - * - * @PREAMBLE strings read from the database files. - */ - buildInFunctions.put("preamble$", context -> { - stack.push(preamble); - }); - - /* - * Pops the top (string) literal, removes nonalphanumeric characters - * except for white-space characters and hyphens and ties (these all get - * converted to a space), removes certain alphabetic characters - * contained in the control sequences associated with a \special - * character", and pushes the resulting string. - */ - buildInFunctions.put("purify$", new PurifyFunction(this)); - - /* - * Pushes the string consisting of the double-quote character. - */ - buildInFunctions.put("quote$", context -> stack.push("\"")); - - /* - * Is a no-op. - */ - buildInFunctions.put("skip$", context -> { - // Nothing to do! Yeah! - }); - - /* - * Pops and prints the whole stack; it's meant to be used for style - * designers while debugging. - */ - buildInFunctions.put("stack$", context -> { - while (!stack.empty()) { - LOGGER.debug("Stack entry {}", stack.pop()); - } - }); - - /* - * Pops the top three literals (they are the two integers literals - * len and start, and a string literal, in that order). It pushes - * the substring of the (at most) len consecutive characters - * starting at the startth character (assuming 1-based indexing) if - * start is positive, and ending at the start-th character - * (including) from the end if start is negative (where the first - * character from the end is the last character). - */ - buildInFunctions.put("substring$", context -> substringFunction()); - - /* - * Swaps the top two literals on the stack. text.length$ Pops the - * top (string) literal, and pushes the number of text characters - * it contains, where an accented character (more precisely, a - * \special character", defined in Section 4) counts as a single - * text character, even if it's missing its matching right brace, - * and where braces don't count as text characters. - */ - buildInFunctions.put("swap$", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation swap$"); - } - Object f1 = stack.pop(); - Object f2 = stack.pop(); - - stack.push(f1); - stack.push(f2); - }); - - /* - * text.length$ Pops the top (string) literal, and pushes the number - * of text characters it contains, where an accented character (more - * precisely, a "special character", defined in Section 4) counts as - * a single text character, even if it's missing its matching right - * brace, and where braces don't count as text characters. - * - * From BibTeXing: For the purposes of counting letters in labels, - * BibTEX considers everything contained inside the braces as a - * single letter. - */ - buildInFunctions.put("text.length$", context -> textLengthFunction()); - - /* - * Pops the top two literals (the integer literal len and a string - * literal, in that order). It pushes the substring of the (at most) len - * consecutive text characters starting from the beginning of the - * string. This function is similar to substring$, but this one - * considers a \special character", even if it's missing its matching - * right brace, to be a single text character (rather than however many - * ASCII characters it actually comprises), and this function doesn't - * consider braces to be text characters; furthermore, this function - * appends any needed matching right braces. - */ - buildInFunctions.put("text.prefix$", new TextPrefixFunction(this)); - - /* - * Pops and prints the top of the stack to the log file. It's useful for debugging. - */ - buildInFunctions.put("top$", context -> LOGGER.debug("Stack entry {}", stack.pop())); - - /* - * Pushes the current entry's type (book, article, etc.), but pushes - * the null string if the type is either unknown or undefined. - */ - buildInFunctions.put("type$", context -> { - if (context == null) { - throw new VMException("type$ need a context."); - } - - stack.push(context.entry.getType().getName()); - }); - - /* - * Pops the top (string) literal and prints it following a warning - * message. This also increments a count of the number of warning - * messages issued. - */ - buildInFunctions.put("warning$", new BstFunction() { - int warning = 1; - - @Override - public void execute(BstEntry context) { - LOGGER.warn("Warning (#" + (warning++) + "): " + stack.pop()); - } - }); - - /* - * Pops the top two (function) literals, and keeps executing the - * second as long as the (integer) literal left on the stack by - * executing the first is greater than 0. - */ - buildInFunctions.put("while$", this::whileFunction); - - buildInFunctions.put("width$", new WidthFunction(this)); - - /* - * Pops the top (string) literal and writes it on the output buffer - * (which will result in stuff being written onto the bbl file when - * the buffer fills up). - */ - buildInFunctions.put("write$", context -> { - String s = (String) stack.pop(); - VM.this.bbl.append(s); - }); - } - - private void textLengthFunction() { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation text.length$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - throw new VMException("Can only perform operation on a string text.length$"); - } - - String s = (String) o1; - char[] c = s.toCharArray(); - int result = 0; - - // Comments from bibtex.web: - - // sp_ptr := str_start[pop_lit1]; - int i = 0; - - // sp_end := str_start[pop_lit1+1]; - int n = s.length(); - - // sp_brace_level := 0; - int braceLevel = 0; - - // while (sp_ptr < sp_end) do begin - while (i < n) { - // incr(sp_ptr); - i++; - // if (str_pool[sp_ptr-1] = left_brace) then - // begin - if (c[i - 1] == '{') { - // incr(sp_brace_level); - braceLevel++; - // if ((sp_brace_level = 1) and (sp_ptr < sp_end)) then - if ((braceLevel == 1) && (i < n)) { - // if (str_pool[sp_ptr] = backslash) then - // begin - if (c[i] == '\\') { - // incr(sp_ptr); {skip over the |backslash|} - i++; // skip over backslash - // while ((sp_ptr < sp_end) and (sp_brace_level - // > 0)) do begin - while ((i < n) && (braceLevel > 0)) { - // if (str_pool[sp_ptr] = right_brace) then - if (c[i] == '}') { - // decr(sp_brace_level) - braceLevel--; - } else if (c[i] == '{') { - // incr(sp_brace_level); - braceLevel++; - } - // incr(sp_ptr); - i++; - // end; - } - // incr(num_text_chars); - result++; - // end; - } - // end - } - - // else if (str_pool[sp_ptr-1] = right_brace) then - // begin - } else if (c[i - 1] == '}') { - // if (sp_brace_level > 0) then - if (braceLevel > 0) { - // decr(sp_brace_level); - braceLevel--; - // end - } - } else { // else - // incr(num_text_chars); - result++; - } - } - stack.push(result); - } - - private void whileFunction(BstEntry context) { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation while$"); - } - Object f2 = stack.pop(); - Object f1 = stack.pop(); - - if (!((f1 instanceof Identifier) || (f1 instanceof Tree)) - && ((f2 instanceof Identifier) || (f2 instanceof Tree))) { - throw new VMException("Expecting two functions for while$."); - } - - do { - VM.this.executeInContext(f1, context); - - Object i = stack.pop(); - if (!(i instanceof Integer)) { - throw new VMException("First parameter to while has to return an integer but was " + i); - } - if ((Integer) i <= 0) { - break; - } - VM.this.executeInContext(f2, context); - } while (true); - } - - private void substringFunction() { - if (stack.size() < 3) { - throw new VMException("Not enough operands on stack for operation substring$"); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - Object o3 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer) && (o3 instanceof String))) { - throw new VMException("Expecting two integers and a string for substring$"); - } - - Integer len = (Integer) o1; - Integer start = (Integer) o2; - - int lenI = len; - int startI = start; - - if (lenI > (Integer.MAX_VALUE / 2)) { - lenI = Integer.MAX_VALUE / 2; - } - - if (startI > (Integer.MAX_VALUE / 2)) { - startI = Integer.MAX_VALUE / 2; - } - - if (startI < (Integer.MIN_VALUE / 2)) { - startI = -Integer.MIN_VALUE / 2; - } - - String s = (String) o3; - - if (startI < 0) { - startI += s.length() + 1; - startI = Math.max(1, (startI + 1) - lenI); - } - stack.push(s.substring(startI - 1, Math.min((startI - 1) + lenI, s.length()))); - } - - private void addPeriodFunction() { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation add.period$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - throw new VMException("Can only add a period to a string for add.period$"); - } - - String s = (String) o1; - Matcher m = ADD_PERIOD_PATTERN.matcher(s); - - if (m.find()) { - StringBuilder sb = new StringBuilder(); - m.appendReplacement(sb, m.group(1)); - sb.append('.'); - String group2 = m.group(2); - if (group2 != null) { - sb.append(m.group(2)); - } - stack.push(sb.toString()); - } else { - stack.push(s); - } - } - - private static CommonTree charStream2CommonTree(CharStream bst) throws RecognitionException { - BstLexer lex = new BstLexer(bst); - CommonTokenStream tokens = new CommonTokenStream(lex); - BstParser parser = new BstParser(tokens); - BstParser.program_return r = parser.program(); - return (CommonTree) r.getTree(); - } - - private boolean assign(BstEntry context, Object o1, Object o2) { - if (!(o1 instanceof Identifier) || !((o2 instanceof String) || (o2 instanceof Integer))) { - throw new VMException("Invalid parameters"); - } - - String name = ((Identifier) o1).getName(); - - if (o2 instanceof String) { - if ((context != null) && context.localStrings.containsKey(name)) { - context.localStrings.put(name, (String) o2); - return true; - } - - if (strings.containsKey(name)) { - strings.put(name, (String) o2); - return true; - } - return false; - } - - if ((context != null) && context.localIntegers.containsKey(name)) { - context.localIntegers.put(name, (Integer) o2); - return true; - } - - if (integers.containsKey(name)) { - integers.put(name, (Integer) o2); - return true; - } - return false; - } - - public String run(BibDatabase db) { - preamble = db.getPreamble().orElse(""); - return run(db.getEntries()); - } - - public String run(Collection bibtex) { - return this.run(bibtex, null); - } - - /** - * Transforms the given list of BibEntries to a rendered list of references using the underlying bst file - * - * @param bibEntries list of entries to convert - * @param bibDatabase (may be null) the bibDatabase used for resolving strings / crossref - * @return list of references in plain text form - */ - public String run(Collection bibEntries, BibDatabase bibDatabase) { - Objects.requireNonNull(bibEntries); - - // Reset - bbl = new StringBuilder(); - - strings = new HashMap<>(); - - integers = new HashMap<>(); - integers.put("entry.max$", Integer.MAX_VALUE); - integers.put("global.max$", Integer.MAX_VALUE); - - functions = new HashMap<>(); - functions.putAll(buildInFunctions); - - stack = new Stack<>(); - - // Create entries - entries = new ArrayList<>(bibEntries.size()); - for (BibEntry entry : bibEntries) { - entries.add(new BstEntry(entry)); - } - - // Go - for (int i = 0; i < tree.getChildCount(); i++) { - Tree child = tree.getChild(i); - switch (child.getType()) { - case BstParser.STRINGS: - strings(child); - break; - case BstParser.INTEGERS: - integers(child); - break; - case BstParser.FUNCTION: - function(child); - break; - case BstParser.EXECUTE: - execute(child); - break; - case BstParser.SORT: - sort(); - break; - case BstParser.ITERATE: - iterate(child); - break; - case BstParser.REVERSE: - reverse(child); - break; - case BstParser.ENTRY: - entry(child); - break; - case BstParser.READ: - read(bibDatabase); - break; - case BstParser.MACRO: - macro(child); - break; - default: - LOGGER.info("Unknown type: {}", child.getType()); - break; - } - } - - return bbl.toString(); - } - - /** - * Dredges up from the database file the field values for each entry in the list. It has no arguments. If a database - * entry doesn't have a value for a field (and probably no database entry will have a value for every field), that - * field variable is marked as missing for the entry. - * - * We use null for the missing entry designator. - */ - private void read(BibDatabase bibDatabase) { - FieldWriter fieldWriter = new FieldWriter(new FieldWriterPreferences(true, List.of(StandardField.MONTH), new FieldContentFormatterPreferences())); - for (BstEntry e : entries) { - for (Map.Entry mEntry : e.fields.entrySet()) { - Field field = FieldFactory.parseField(mEntry.getKey()); - String fieldValue = e.entry.getResolvedFieldOrAlias(field, bibDatabase) - .map(content -> { - try { - String result = fieldWriter.write(field, content); - if (result.startsWith("{")) { - // Strip enclosing {} from the output - return result.substring(1, result.length() - 1); - } - if (field == StandardField.MONTH) { - // We don't have the internal BibTeX strings at hand. - // We nevertheless want to have the full month name. - // Thus, we lookup the full month name here. - return Month.parse(result) - .map(month -> month.getFullName()) - .orElse(result); - } - return result; - } catch (InvalidFieldValueException invalidFieldValueException) { - // in case there is something wrong with the content, just return the content itself - return content; - } - }) - .orElse(null); - mEntry.setValue(fieldValue); - } - } - - for (BstEntry e : entries) { - if (!e.fields.containsKey(StandardField.CROSSREF.getName())) { - e.fields.put(StandardField.CROSSREF.getName(), null); - } - } - } - - /** - * Defines a string macro. It has two arguments; the first is the macro's name, which is treated like any other - * variable or function name, and the second is its definition, which must be double-quote-delimited. You must have - * one for each three-letter month abbreviation; in addition, you should have one for common journal names. The - * user's database may override any definition you define using this command. If you want to define a string the - * user can't touch, use the FUNCTION command, which has a compatible syntax. - */ - private void macro(Tree child) { - String name = child.getChild(0).getText(); - String replacement = child.getChild(1).getText(); - functions.put(name, new MacroFunction(replacement)); - } - - public class MacroFunction implements BstFunction { - - private final String replacement; - - public MacroFunction(String replacement) { - this.replacement = replacement; - } - - @Override - public void execute(BstEntry context) { - VM.this.push(replacement); - } - } - - /** - * Declares the fields and entry variables. It has three arguments, each a (possibly empty) list of variable names. - * The three lists are of: fields, integer entry variables, and string entry variables. There is an additional field - * that BibTEX automatically declares, crossref, used for cross referencing. And there is an additional string entry - * variable automatically declared, sort.key$, used by the SORT command. Each of these variables has a value for - * each entry on the list. - */ - private void entry(Tree child) { - // Fields first - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - - for (BstEntry entry : entries) { - entry.fields.put(name, null); - } - } - - // Integers - t = child.getChild(1); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - - for (BstEntry entry : entries) { - entry.localIntegers.put(name, 0); - } - } - // Strings - t = child.getChild(2); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - for (BstEntry entry : entries) { - entry.localStrings.put(name, null); - } - } - for (BstEntry entry : entries) { - entry.localStrings.put("sort.key$", null); - } - } - - private void reverse(Tree child) { - BstFunction f = functions.get(child.getChild(0).getText()); - - ListIterator i = entries.listIterator(entries.size()); - while (i.hasPrevious()) { - f.execute(i.previous()); - } - } - - private void iterate(Tree child) { - BstFunction f = functions.get(child.getChild(0).getText()); - - for (BstEntry entry : entries) { - f.execute(entry); - } - } - - /** - * Sorts the entry list using the values of the string entry variable sort.key$. It has no arguments. - */ - private void sort() { - entries.sort(Comparator.comparing(o -> (o.localStrings.get("sort.key$")))); - } - - private void executeInContext(Object o, BstEntry context) { - if (o instanceof Tree) { - Tree t = (Tree) o; - new StackFunction(t).execute(context); - } else if (o instanceof Identifier) { - execute(((Identifier) o).getName(), context); - } - } - - private void execute(Tree child) { - execute(child.getChild(0).getText(), null); - } - - public class StackFunction implements BstFunction { - - private final Tree localTree; - - public StackFunction(Tree stack) { - localTree = stack; - } - - public Tree getTree() { - return localTree; - } - - @Override - public void execute(BstEntry context) { - for (int i = 0; i < localTree.getChildCount(); i++) { - Tree c = localTree.getChild(i); - try { - - switch (c.getType()) { - case BstParser.STRING: - String s = c.getText(); - push(s.substring(1, s.length() - 1)); - break; - case BstParser.INTEGER: - push(Integer.parseInt(c.getText().substring(1))); - break; - case BstParser.QUOTED: - push(new Identifier(c.getText().substring(1))); - break; - case BstParser.STACK: - push(c); - break; - default: - VM.this.execute(c.getText(), context); - break; - } - } catch (VMException e) { - if (file == null) { - LOGGER.error("ERROR " + e.getMessage() + " (" + c.getLine() + ")"); - } else { - LOGGER.error("ERROR " + e.getMessage() + " (" + file.getPath() + ":" - + c.getLine() + ")"); - } - throw e; - } - } - } - } - - private void push(Tree t) { - stack.push(t); - } - - private void execute(String name, BstEntry context) { - if (context != null) { - if (context.fields.containsKey(name)) { - stack.push(context.fields.get(name)); - return; - } - if (context.localStrings.containsKey(name)) { - stack.push(context.localStrings.get(name)); - return; - } - if (context.localIntegers.containsKey(name)) { - stack.push(context.localIntegers.get(name)); - return; - } - } - if (strings.containsKey(name)) { - stack.push(strings.get(name)); - return; - } - if (integers.containsKey(name)) { - stack.push(integers.get(name)); - return; - } - - if (functions.containsKey(name)) { - // OK to have a null context - functions.get(name).execute(context); - return; - } - - throw new VMException("No matching identifier found: " + name); - } - - private void function(Tree child) { - String name = child.getChild(0).getText(); - Tree localStack = child.getChild(1); - functions.put(name, new StackFunction(localStack)); - } - - /** - * Declares global integer variables. It has one argument, a list of variable names. There are two such - * automatically-declared variables, entry.max$ and global.max$, used for limiting the lengths of string vari- - * ables. You may have any number of these commands, but a variable's declaration must precede its use. - */ - private void integers(Tree child) { - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - integers.put(name, 0); - } - } - - /** - * Declares global string variables. It has one argument, a list of variable names. You may have any number of these - * commands, but a variable's declaration must precede its use. - * - * @param child - */ - private void strings(Tree child) { - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - strings.put(name, null); - } - } - - public static class BstEntry { - - public final BibEntry entry; - - public final Map localStrings = new HashMap<>(); - - // keys filled by org.jabref.logic.bst.VM.entry based on the contents of the bst file - public final Map fields = new HashMap<>(); - - public final Map localIntegers = new HashMap<>(); - - public BstEntry(BibEntry e) { - this.entry = e; - } - } - - private void push(Integer integer) { - stack.push(integer); - } - - private void push(String string) { - stack.push(string); - } - - private void push(Identifier identifier) { - stack.push(identifier); - } - - public Map getStrings() { - return strings; - } - - public Map getIntegers() { - return integers; - } - - public List getEntries() { - return entries; - } - - public Map getFunctions() { - return functions; - } - - public Stack getStack() { - return stack; - } - - @Override - public void warn(String string) { - LOGGER.warn(string); - } -} diff --git a/src/main/java/org/jabref/logic/bst/VMException.java b/src/main/java/org/jabref/logic/bst/VMException.java deleted file mode 100644 index f46c5cd5277..00000000000 --- a/src/main/java/org/jabref/logic/bst/VMException.java +++ /dev/null @@ -1,8 +0,0 @@ -package org.jabref.logic.bst; - -public class VMException extends RuntimeException { - - public VMException(String string) { - super(string); - } -} diff --git a/src/main/java/org/jabref/logic/bst/Warn.java b/src/main/java/org/jabref/logic/bst/Warn.java deleted file mode 100644 index 7a524ad9834..00000000000 --- a/src/main/java/org/jabref/logic/bst/Warn.java +++ /dev/null @@ -1,7 +0,0 @@ -package org.jabref.logic.bst; - -@FunctionalInterface -public interface Warn { - - void warn(String s); -} diff --git a/src/main/java/org/jabref/logic/bst/WidthFunction.java b/src/main/java/org/jabref/logic/bst/WidthFunction.java deleted file mode 100644 index 06784267bf0..00000000000 --- a/src/main/java/org/jabref/logic/bst/WidthFunction.java +++ /dev/null @@ -1,43 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Stack; - -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; - -/** - * The |built_in| function {\.{width\$}} pops the top (string) literal and - * pushes the integer that represents its width in units specified by the - * |char_width| array. This function takes the literal literally; that is, it - * assumes each character in the string is to be printed as is, regardless of - * whether the character has a special meaning to \TeX, except that special - * characters (even without their |right_brace|s) are handled specially. If the - * literal isn't a string, it complains and pushes~0. - * - */ -public class WidthFunction implements BstFunction { - - private final VM vm; - - public WidthFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation width$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - vm.warn("A string is needed for change.case$"); - stack.push(0); - return; - } - - stack.push(BibtexWidth.width((String) o1)); - } -} diff --git a/src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java b/src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java similarity index 83% rename from src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java rename to src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java index e05ce998e80..dd3e7c7b05c 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java +++ b/src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java @@ -1,4 +1,4 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; import java.util.Locale; import java.util.Optional; @@ -6,9 +6,9 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; -public final class BibtexCaseChanger { +public final class BstCaseChanger { - private static final Logger LOGGER = LoggerFactory.getLogger(BibtexCaseChanger.class); + private static final Logger LOGGER = LoggerFactory.getLogger(BstCaseChanger.class); // stores whether the char before the current char was a colon private boolean prevColon = true; @@ -16,7 +16,7 @@ public final class BibtexCaseChanger { // global variable to store the current brace level private int braceLevel; - public enum FORMAT_MODE { + public enum FormatMode { // First character and character after a ":" as upper case - everything else in lower case. Obey {}. TITLE_LOWERS('t'), @@ -40,7 +40,7 @@ public enum FORMAT_MODE { private final char asChar; - FORMAT_MODE(char asChar) { + FormatMode(char asChar) { this.asChar = asChar; } @@ -53,17 +53,21 @@ public char asChar() { * * @throws IllegalArgumentException if char is not 't', 'l', 'u' */ - public static FORMAT_MODE getFormatModeForBSTFormat(final char bstFormat) { - for (FORMAT_MODE mode : FORMAT_MODE.values()) { + public static FormatMode of(final char bstFormat) { + for (FormatMode mode : FormatMode.values()) { if (mode.asChar == bstFormat) { return mode; } } throw new IllegalArgumentException(); } + + public static FormatMode of(final String bstFormat) { + return of(bstFormat.toLowerCase(Locale.ROOT).charAt(0)); + } } - private BibtexCaseChanger() { + private BstCaseChanger() { } /** @@ -72,11 +76,11 @@ private BibtexCaseChanger() { * @param s the string to handle * @param format the format */ - public static String changeCase(String s, FORMAT_MODE format) { - return (new BibtexCaseChanger()).doChangeCase(s, format); + public static String changeCase(String s, FormatMode format) { + return (new BstCaseChanger()).doChangeCase(s, format); } - private String doChangeCase(String s, FORMAT_MODE format) { + private String doChangeCase(String s, FormatMode format) { char[] c = s.toCharArray(); StringBuilder sb = new StringBuilder(); @@ -93,7 +97,7 @@ private String doChangeCase(String s, FORMAT_MODE format) { i++; continue; } - if ((format == FORMAT_MODE.TITLE_LOWERS) && ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1])))) { + if ((format == FormatMode.TITLE_LOWERS) && ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1])))) { sb.append('{'); i++; prevColon = false; @@ -136,12 +140,9 @@ private String doChangeCase(String s, FORMAT_MODE format) { * is other stuff, too, between braces, but it doesn't try to do anything * special with |colon|s. * - * @param c * @param start the current position. It points to the opening brace - * @param format - * @return */ - private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MODE format) { + private int convertSpecialChar(StringBuilder sb, char[] c, int start, FormatMode format) { int i = start; sb.append(c[i]); @@ -152,7 +153,7 @@ private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MOD i++; // skip over the |backslash| - Optional s = BibtexCaseChanger.findSpecialChar(c, i); + Optional s = BstCaseChanger.findSpecialChar(c, i); if (s.isPresent()) { i = convertAccented(c, i, s.get(), sb, format); } @@ -174,14 +175,9 @@ private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MOD * up) and append the result to the stringBuffer, return the updated * position. * - * @param c - * @param start - * @param s - * @param sb - * @param format * @return the new position */ - private int convertAccented(char[] c, int start, String s, StringBuilder sb, FORMAT_MODE format) { + private int convertAccented(char[] c, int start, String s, StringBuilder sb, FormatMode format) { int pos = start; pos += s.length(); @@ -214,29 +210,27 @@ private int convertAccented(char[] c, int start, String s, StringBuilder sb, FOR return pos; } - private int convertNonControl(char[] c, int start, StringBuilder sb, FORMAT_MODE format) { + private int convertNonControl(char[] c, int start, StringBuilder sb, FormatMode format) { int pos = start; switch (format) { - case TITLE_LOWERS: - case ALL_LOWERS: + case TITLE_LOWERS, ALL_LOWERS -> { sb.append(Character.toLowerCase(c[pos])); pos++; - break; - case ALL_UPPERS: + } + case ALL_UPPERS -> { sb.append(Character.toUpperCase(c[pos])); pos++; - break; - default: - LOGGER.info("convertNonControl - Unknown format: " + format); - break; + } + default -> + LOGGER.info("convertNonControl - Unknown format: " + format); } return pos; } - private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, FORMAT_MODE format) { + private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, FormatMode format) { int i = start; switch (format) { - case TITLE_LOWERS: + case TITLE_LOWERS -> { if ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1]))) { sb.append(c[i]); } else { @@ -247,16 +241,13 @@ private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, } else if (!Character.isWhitespace(c[i])) { prevColon = false; } - break; - case ALL_LOWERS: - sb.append(Character.toLowerCase(c[i])); - break; - case ALL_UPPERS: - sb.append(Character.toUpperCase(c[i])); - break; - default: - LOGGER.info("convertCharIfBraceLevelIsZero - Unknown format: " + format); - break; + } + case ALL_LOWERS -> + sb.append(Character.toLowerCase(c[i])); + case ALL_UPPERS -> + sb.append(Character.toUpperCase(c[i])); + default -> + LOGGER.info("convertCharIfBraceLevelIsZero - Unknown format: " + format); } i++; return i; diff --git a/src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java b/src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java similarity index 76% rename from src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java rename to src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java index aecf571bdf4..c4986dc12dd 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java +++ b/src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java @@ -1,13 +1,17 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; import java.util.Arrays; import java.util.Locale; import java.util.Optional; import java.util.stream.Collectors; +import org.jabref.logic.bst.BstVMException; import org.jabref.model.entry.Author; import org.jabref.model.entry.AuthorList; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + /** * From Bibtex: * @@ -24,9 +28,10 @@ * Sounds easy - is a nightmare... X-( * */ -public class BibtexNameFormatter { +public class BstNameFormatter { + private static final Logger LOGGER = LoggerFactory.getLogger(BstNameFormatter.class); - private BibtexNameFormatter() { + private BstNameFormatter() { } /** @@ -35,23 +40,18 @@ private BibtexNameFormatter() { * @param authorsNameList The string from an author field * @param whichName index of the list, starting with 1 * @param formatString TODO - * @param warn collects the warnings, may-be-null - * @return */ - public static String formatName(String authorsNameList, int whichName, String formatString, Warn warn) { + public static String formatName(String authorsNameList, int whichName, String formatString) { AuthorList al = AuthorList.parse(authorsNameList); if ((whichName < 1) && (whichName > al.getNumberOfAuthors())) { - warn.warn("AuthorList " + authorsNameList + " does not contain an author with number " + whichName); + LOGGER.warn("AuthorList {} does not contain an author with number {}", authorsNameList, whichName); return ""; } - return BibtexNameFormatter.formatName(al.getAuthor(whichName - 1), formatString, warn); + return BstNameFormatter.formatName(al.getAuthor(whichName - 1), formatString); } - /** - * @param warn collects the warnings, may-be-null - */ - public static String formatName(Author author, String format, Warn warn) { + public static String formatName(Author author, String format) { StringBuilder sb = new StringBuilder(); char[] c = format.toCharArray(); @@ -81,11 +81,7 @@ public static String formatName(Author author, String format, Warn warn) { } if ((braceLevel == 1) && Character.isLetter(c[i])) { if ("fvlj".indexOf(c[i]) == -1) { - if (warn != null) { - warn.warn( - "Format string in format.name$ may only contain fvlj on brace level 1 in group " - + group + ": " + format); - } + LOGGER.warn("Format string in format.name$ may only contain fvlj on brace level 1 in group {}: {}", group, format); } else { level1Chars.append(c[i]); } @@ -99,31 +95,26 @@ public static String formatName(Author author, String format, Warn warn) { continue; } - if ((control.length() > 2) && (warn != null)) { - warn.warn("Format string in format.name$ may only be one or two character long on brace level 1 in group " + group + ": " + format); + if ((control.length() > 2)) { + LOGGER.warn("Format string in format.name$ may only be one or two character long on brace level 1 in group {}: {}", group, format); } char type = control.charAt(0); - Optional tokenS; - switch (type) { - case 'f': - tokenS = author.getFirst(); - break; - case 'v': - tokenS = author.getVon(); - break; - case 'l': - tokenS = author.getLast(); - break; - case 'j': - tokenS = author.getJr(); - break; - default: - throw new VMException("Internal error"); - } - - if (!tokenS.isPresent()) { + Optional tokenS = switch (type) { + case 'f' -> + author.getFirst(); + case 'v' -> + author.getVon(); + case 'l' -> + author.getLast(); + case 'j' -> + author.getJr(); + default -> + throw new BstVMException("Internal error"); + }; + + if (tokenS.isEmpty()) { i++; continue; } @@ -135,9 +126,7 @@ public static String formatName(Author author, String format, Warn warn) { if (control.charAt(1) == control.charAt(0)) { abbreviateThatIsSingleLetter = false; } else { - if (warn != null) { - warn.warn("Format string in format.name$ may only contain one type of vlfj on brace level 1 in group " + group + ": " + format); - } + LOGGER.warn("Format string in format.name$ may only contain one type of vlfj on brace level 1 in group {}: {}", group, format); } } @@ -162,7 +151,7 @@ public static String formatName(Author author, String format, Warn warn) { } if (((j + 1) < d.length) && (d[j + 1] == '{')) { StringBuilder interTokenSb = new StringBuilder(); - j = BibtexNameFormatter.consumeToMatchingBrace(interTokenSb, d, j + 1); + j = BstNameFormatter.consumeToMatchingBrace(interTokenSb, d, j + 1); interToken = interTokenSb.substring(1, interTokenSb.length() - 1); } @@ -171,7 +160,7 @@ public static String formatName(Author author, String format, Warn warn) { if (abbreviateThatIsSingleLetter) { String[] dashes = token.split("-"); - token = Arrays.asList(dashes).stream().map(BibtexNameFormatter::getFirstCharOfString) + token = Arrays.stream(dashes).map(BstNameFormatter::getFirstCharOfString) .collect(Collectors.joining(".-")); } @@ -187,7 +176,7 @@ public static String formatName(Author author, String format, Warn warn) { // No clue what this means (What the hell are tokens anyway??? // if (lex_class[name_sep_char[cur_token]] = sep_char) then // append_ex_buf_char_and_check (name_sep_char[cur_token]) - if ((k == (tokens.length - 2)) || (BibtexNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 3) < 3)) { + if ((k == (tokens.length - 2)) || (BstNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 3) < 3)) { sb.append('~'); } else { sb.append(' '); @@ -212,7 +201,7 @@ public static String formatName(Author author, String format, Warn warn) { if (sb.length() > 0) { boolean noDisTie = false; if ((sb.charAt(sb.length() - 1) == '~') && - ((BibtexNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 4) >= 4) || + ((BstNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 4) >= 4) || ((sb.length() > 1) && (noDisTie = sb.charAt(sb.length() - 2) == '~')))) { sb.deleteCharAt(sb.length() - 1); if (!noDisTie) { @@ -221,16 +210,14 @@ public static String formatName(Author author, String format, Warn warn) { } } } else if (c[i] == '}') { - if (warn != null) { - warn.warn("Unmatched brace in format string: " + format); - } + LOGGER.warn("Unmatched brace in format string: {}", format); } else { sb.append(c[i]); // verbatim } i++; } - if ((braceLevel != 0) && (warn != null)) { - warn.warn("Unbalanced brace in format string for nameFormat: " + format); + if ((braceLevel != 0)) { + LOGGER.warn("Unbalanced brace in format string for nameFormat: {}", format); } return sb.toString(); @@ -268,7 +255,7 @@ public static String getFirstCharOfString(String s) { } if ((c[i] == '{') && ((i + 1) < c.length) && (c[i + 1] == '\\')) { StringBuilder sb = new StringBuilder(); - BibtexNameFormatter.consumeToMatchingBrace(sb, c, i); + BstNameFormatter.consumeToMatchingBrace(sb, c, i); return sb.toString(); } } diff --git a/src/main/java/org/jabref/logic/bst/BibtexPurify.java b/src/main/java/org/jabref/logic/bst/util/BstPurifier.java similarity index 77% rename from src/main/java/org/jabref/logic/bst/BibtexPurify.java rename to src/main/java/org/jabref/logic/bst/util/BstPurifier.java index 08cb14db207..90818a0749f 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexPurify.java +++ b/src/main/java/org/jabref/logic/bst/util/BstPurifier.java @@ -1,4 +1,7 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * @@ -10,17 +13,13 @@ * pushes the null string. * */ -public class BibtexPurify { +public class BstPurifier { + private static final Logger LOGGER = LoggerFactory.getLogger(BstPurifier.class); - private BibtexPurify() { + private BstPurifier() { } - /** - * @param toPurify - * @param warn may-be-null - * @return - */ - public static String purify(String toPurify, Warn warn) { + public static String purify(String toPurify) { StringBuilder sb = new StringBuilder(); char[] cs = toPurify.toCharArray(); @@ -41,7 +40,7 @@ public static String purify(String toPurify, Warn warn) { i++; // skip brace while ((i < n) && (braceLevel > 0)) { i++; // skip backslash - BibtexCaseChanger.findSpecialChar(cs, i).ifPresent(sb::append); + BstCaseChanger.findSpecialChar(cs, i).ifPresent(sb::append); while ((i < n) && Character.isLetter(cs[i])) { i++; @@ -63,15 +62,13 @@ public static String purify(String toPurify, Warn warn) { if (braceLevel > 0) { braceLevel--; } else { - if (warn != null) { - warn.warn("Unbalanced brace in string for purify$: " + toPurify); - } + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPurify); } } i++; } - if ((braceLevel != 0) && (warn != null)) { - warn.warn("Unbalanced brace in string for purify$: " + toPurify); + if ((braceLevel != 0)) { + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPurify); } return sb.toString(); diff --git a/src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java b/src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java similarity index 84% rename from src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java rename to src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java index 721838e29fe..31cbc71de9b 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java +++ b/src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java @@ -1,4 +1,7 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * The |built_in| function {\.{text.prefix\$}} pops the top two literals (the @@ -14,15 +17,13 @@ * complains and pushes the null string. * */ -public class BibtexTextPrefix { +public class BstTextPrefixer { + private static final Logger LOGGER = LoggerFactory.getLogger(BstTextPrefixer.class); - private BibtexTextPrefix() { + private BstTextPrefixer() { } - /** - * @param warn may-be-null - */ - public static String textPrefix(int inNumOfChars, String toPrefix, Warn warn) { + public static String textPrefix(int inNumOfChars, String toPrefix) { int numOfChars = inNumOfChars; StringBuilder sb = new StringBuilder(); @@ -53,15 +54,13 @@ public static String textPrefix(int inNumOfChars, String toPrefix, Warn warn) { if (braceLevel > 0) { braceLevel--; } else { - if (warn != null) { - warn.warn("Unbalanced brace in string for purify$: " + toPrefix); - } + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPrefix); } } else { numOfChars--; } } - sb.append(toPrefix.substring(0, i)); + sb.append(toPrefix, 0, i); while (braceLevel > 0) { sb.append('}'); braceLevel--; diff --git a/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java b/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java new file mode 100644 index 00000000000..c286fe1497b --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java @@ -0,0 +1,241 @@ +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * + * The |built_in| function {\.{purify\$}} pops the top (string) literal, removes + * nonalphanumeric characters except for |white_space| and |sep_char| characters + * (these get converted to a |space|) and removes certain alphabetic characters + * contained in the control sequences associated with a special character, and + * pushes the resulting string. If the literal isn't a string, it complains and + * pushes the null string. + * + */ +public class BstWidthCalculator { + + private static final Logger LOGGER = LoggerFactory.getLogger(BstWidthCalculator.class); + + /* + * Quoted from Bibtex: + * + * Now we initialize the system-dependent |char_width| array, for which + * |space| is the only |white_space| character given a nonzero printing + * width. The widths here are taken from Stanford's June~'87 $cmr10$~font + * and represent hundredths of a point (rounded), but since they're used + * only for relative comparisons, the units have no meaning. + */ + + private static int[] widths; + + static { + if (BstWidthCalculator.widths == null) { + BstWidthCalculator.widths = new int[128]; + + for (int i = 0; i < 128; i++) { + BstWidthCalculator.widths[i] = 0; + } + BstWidthCalculator.widths[32] = 278; + BstWidthCalculator.widths[33] = 278; + BstWidthCalculator.widths[34] = 500; + BstWidthCalculator.widths[35] = 833; + BstWidthCalculator.widths[36] = 500; + BstWidthCalculator.widths[37] = 833; + BstWidthCalculator.widths[38] = 778; + BstWidthCalculator.widths[39] = 278; + BstWidthCalculator.widths[40] = 389; + BstWidthCalculator.widths[41] = 389; + BstWidthCalculator.widths[42] = 500; + BstWidthCalculator.widths[43] = 778; + BstWidthCalculator.widths[44] = 278; + BstWidthCalculator.widths[45] = 333; + BstWidthCalculator.widths[46] = 278; + BstWidthCalculator.widths[47] = 500; + BstWidthCalculator.widths[48] = 500; + BstWidthCalculator.widths[49] = 500; + BstWidthCalculator.widths[50] = 500; + BstWidthCalculator.widths[51] = 500; + BstWidthCalculator.widths[52] = 500; + BstWidthCalculator.widths[53] = 500; + BstWidthCalculator.widths[54] = 500; + BstWidthCalculator.widths[55] = 500; + BstWidthCalculator.widths[56] = 500; + BstWidthCalculator.widths[57] = 500; + BstWidthCalculator.widths[58] = 278; + BstWidthCalculator.widths[59] = 278; + BstWidthCalculator.widths[60] = 278; + BstWidthCalculator.widths[61] = 778; + BstWidthCalculator.widths[62] = 472; + BstWidthCalculator.widths[63] = 472; + BstWidthCalculator.widths[64] = 778; + BstWidthCalculator.widths[65] = 750; + BstWidthCalculator.widths[66] = 708; + BstWidthCalculator.widths[67] = 722; + BstWidthCalculator.widths[68] = 764; + BstWidthCalculator.widths[69] = 681; + BstWidthCalculator.widths[70] = 653; + BstWidthCalculator.widths[71] = 785; + BstWidthCalculator.widths[72] = 750; + BstWidthCalculator.widths[73] = 361; + BstWidthCalculator.widths[74] = 514; + BstWidthCalculator.widths[75] = 778; + BstWidthCalculator.widths[76] = 625; + BstWidthCalculator.widths[77] = 917; + BstWidthCalculator.widths[78] = 750; + BstWidthCalculator.widths[79] = 778; + BstWidthCalculator.widths[80] = 681; + BstWidthCalculator.widths[81] = 778; + BstWidthCalculator.widths[82] = 736; + BstWidthCalculator.widths[83] = 556; + BstWidthCalculator.widths[84] = 722; + BstWidthCalculator.widths[85] = 750; + BstWidthCalculator.widths[86] = 750; + BstWidthCalculator.widths[87] = 1028; + BstWidthCalculator.widths[88] = 750; + BstWidthCalculator.widths[89] = 750; + BstWidthCalculator.widths[90] = 611; + BstWidthCalculator.widths[91] = 278; + BstWidthCalculator.widths[92] = 500; + BstWidthCalculator.widths[93] = 278; + BstWidthCalculator.widths[94] = 500; + BstWidthCalculator.widths[95] = 278; + BstWidthCalculator.widths[96] = 278; + BstWidthCalculator.widths[97] = 500; + BstWidthCalculator.widths[98] = 556; + BstWidthCalculator.widths[99] = 444; + BstWidthCalculator.widths[100] = 556; + BstWidthCalculator.widths[101] = 444; + BstWidthCalculator.widths[102] = 306; + BstWidthCalculator.widths[103] = 500; + BstWidthCalculator.widths[104] = 556; + BstWidthCalculator.widths[105] = 278; + BstWidthCalculator.widths[106] = 306; + BstWidthCalculator.widths[107] = 528; + BstWidthCalculator.widths[108] = 278; + BstWidthCalculator.widths[109] = 833; + BstWidthCalculator.widths[110] = 556; + BstWidthCalculator.widths[111] = 500; + BstWidthCalculator.widths[112] = 556; + BstWidthCalculator.widths[113] = 528; + BstWidthCalculator.widths[114] = 392; + BstWidthCalculator.widths[115] = 394; + BstWidthCalculator.widths[116] = 389; + BstWidthCalculator.widths[117] = 556; + BstWidthCalculator.widths[118] = 528; + BstWidthCalculator.widths[119] = 722; + BstWidthCalculator.widths[120] = 528; + BstWidthCalculator.widths[121] = 528; + BstWidthCalculator.widths[122] = 444; + BstWidthCalculator.widths[123] = 500; + BstWidthCalculator.widths[124] = 1000; + BstWidthCalculator.widths[125] = 500; + BstWidthCalculator.widths[126] = 500; + } + } + + private BstWidthCalculator() { + } + + private static int getSpecialCharWidth(char[] c, int pos) { + if ((pos + 1) < c.length) { + if ((c[pos] == 'o') && (c[pos + 1] == 'e')) { + return 778; + } + if ((c[pos] == 'O') && (c[pos + 1] == 'E')) { + return 1014; + } + if ((c[pos] == 'a') && (c[pos + 1] == 'e')) { + return 722; + } + if ((c[pos] == 'A') && (c[pos + 1] == 'E')) { + return 903; + } + if ((c[pos] == 's') && (c[pos + 1] == 's')) { + return 500; + } + } + return BstWidthCalculator.getCharWidth(c[pos]); + } + + public static int getCharWidth(char c) { + if ((c >= 0) && (c < 128)) { + return BstWidthCalculator.widths[c]; + } else { + return 0; + } + } + + public static int width(String toMeasure) { + /* + * From Bibtex: We use the natural width for all but special characters, + * and we complain if the string isn't brace-balanced. + */ + + int i = 0; + int n = toMeasure.length(); + int braceLevel = 0; + char[] c = toMeasure.toCharArray(); + int result = 0; + + /* + * From Bibtex: + * + * We use the natural widths of all characters except that some + * characters have no width: braces, control sequences (except for the + * usual 13 accented and foreign characters, whose widths are given in + * the next module), and |white_space| following control sequences (even + * a null control sequence). + * + */ + while (i < n) { + if (c[i] == '{') { + braceLevel++; + if ((braceLevel == 1) && ((i + 1) < n) && (c[i + 1] == '\\')) { + i++; // skip brace + while ((i < n) && (braceLevel > 0)) { + i++; // skip backslash + + int afterBackslash = i; + while ((i < n) && Character.isLetter(c[i])) { + i++; + } + if ((i < n) && (i == afterBackslash)) { + i++; // Skip non-alpha control seq + } else { + if (BstCaseChanger.findSpecialChar(c, afterBackslash).isPresent()) { + result += BstWidthCalculator.getSpecialCharWidth(c, afterBackslash); + } + } + while ((i < n) && Character.isWhitespace(c[i])) { + i++; + } + while ((i < n) && (braceLevel > 0) && (c[i] != '\\')) { + if (c[i] == '}') { + braceLevel--; + } else if (c[i] == '{') { + braceLevel++; + } else { + result += BstWidthCalculator.getCharWidth(c[i]); + } + i++; + } + } + continue; + } + } else if (c[i] == '}') { + if (braceLevel > 0) { + braceLevel--; + } else { + LOGGER.warn("Too many closing braces in string: " + toMeasure); + } + } + result += BstWidthCalculator.getCharWidth(c[i]); + i++; + } + if (braceLevel > 0) { + LOGGER.warn("No enough closing braces in string: " + toMeasure); + } + return result; + } +} diff --git a/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java b/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java index c0f564800c0..1b98ae4aa15 100644 --- a/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java +++ b/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java @@ -110,7 +110,11 @@ protected void writeEntryTypeDefinition(BibEntryType customType) throws IOExcept @Override protected void writeProlog(BibDatabaseContext bibDatabaseContext, Charset encoding) throws IOException { - if ((encoding == null) || (encoding == StandardCharsets.UTF_8)) { + // We write the encoding if + // - it is provided (!= null) + // - explicitly set in the .bib file OR not equal to UTF_8 + // Otherwise, we do not write anything and return + if ((encoding == null) || (!bibDatabaseContext.getMetaData().getEncodingExplicitlySupplied() && (encoding == StandardCharsets.UTF_8))) { return; } diff --git a/src/main/java/org/jabref/logic/importer/ImportFormatReader.java b/src/main/java/org/jabref/logic/importer/ImportFormatReader.java index 617cd5c0dba..f3b8826171d 100644 --- a/src/main/java/org/jabref/logic/importer/ImportFormatReader.java +++ b/src/main/java/org/jabref/logic/importer/ImportFormatReader.java @@ -195,7 +195,7 @@ public UnknownFormatImport importUnknownFormat(Path filePath, FileUpdateMonitor parserResult.setPath(filePath); return new UnknownFormatImport(ImportFormatReader.BIBTEX_FORMAT, parserResult); } else { - throw new ImportException(Localization.lang("Could not find a suitable import format.")); + throw new ImportException(parserResult.getErrorMessage()); } } catch (IOException ignore) { // Ignored diff --git a/src/main/java/org/jabref/logic/importer/Importer.java b/src/main/java/org/jabref/logic/importer/Importer.java index f6ffc966214..56bde6f0e3e 100644 --- a/src/main/java/org/jabref/logic/importer/Importer.java +++ b/src/main/java/org/jabref/logic/importer/Importer.java @@ -123,12 +123,10 @@ protected static Charset getCharset(BufferedInputStream bufferedInputStream) { if ((matches == null) || (matches.length == 0)) { return defaultCharSet; } - if (Arrays.stream(matches).anyMatch(singleCharset -> singleCharset.getName().equals(defaultCharSet.toString()))) { + + if (Arrays.stream(matches).anyMatch(charset -> "ASCII".equals(charset.getName()))) { return defaultCharSet; } - if (Arrays.stream(matches).anyMatch(singleCharset -> singleCharset.getName().equals(StandardCharsets.UTF_16.toString()))) { - return StandardCharsets.UTF_16; - } if (matches[0] != null) { return Charset.forName(matches[0].getName()); @@ -166,7 +164,7 @@ public static BufferedReader getReader(Path filePath) throws IOException { return new BufferedReader(new InputStreamReader(stream, StandardCharsets.UTF_8)); } - public static BufferedReader getReader(InputStream stream) throws IOException { + public static BufferedReader getReader(InputStream stream) { BufferedInputStream bufferedInputStream = new BufferedInputStream(stream); Charset charset = getCharset(bufferedInputStream); InputStreamReader reader = new InputStreamReader(bufferedInputStream, charset); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java index 5fb3d8ed900..65301e2a7a1 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java @@ -184,7 +184,7 @@ private void parse(T entryType, Map fields) { } else if (isMethodToIgnore(method.getName())) { continue; } else if (method.getName().startsWith("get")) { - putIfValueNotNull(fields, FieldFactory.parseField(method.getName().replace("get", "")), (String) method.invoke(entryType)); + putIfValueNotNull(fields, FieldFactory.parseField(entryType, method.getName().replace("get", "")), (String) method.invoke(entryType)); } } catch (IllegalArgumentException | InvocationTargetException | IllegalAccessException e) { LOGGER.error("Could not invoke method", e); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java index e35fe6c73ad..590ae78de4e 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java @@ -4,7 +4,10 @@ import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; +import java.io.InputStreamReader; import java.nio.charset.Charset; +import java.nio.charset.CharsetDecoder; +import java.nio.charset.CodingErrorAction; import java.nio.file.Files; import java.nio.file.Path; import java.util.Objects; @@ -68,18 +71,28 @@ public ParserResult importDatabase(Path filePath) throws IOException { } Charset encoding; + boolean encodingExplicitlySupplied; try (BufferedReader reader = Files.newBufferedReader(filePath, detectedCharset)) { Optional suppliedEncoding = getSuppliedEncoding(reader); LOGGER.debug("Supplied encoding: {}", suppliedEncoding); + encodingExplicitlySupplied = suppliedEncoding.isPresent(); // in case no encoding information is present, use the detected one encoding = suppliedEncoding.orElse(detectedCharset); LOGGER.debug("Encoding used to read the file: {}", encoding); } - try (BufferedReader reader = Files.newBufferedReader(filePath, encoding)) { + // We replace unreadable characters + // Unfortunately, no warning will be issued to the user + // As this is a very seldom case, we accept that + CharsetDecoder decoder = encoding.newDecoder(); + decoder.onMalformedInput(CodingErrorAction.REPLACE); + + try (InputStreamReader inputStreamReader = new InputStreamReader(Files.newInputStream(filePath), decoder); + BufferedReader reader = new BufferedReader(inputStreamReader)) { ParserResult parserResult = this.importDatabase(reader); parserResult.getMetaData().setEncoding(encoding); + parserResult.getMetaData().setEncodingExplicitlySupplied(encodingExplicitlySupplied); parserResult.setPath(filePath); if (parserResult.getMetaData().getMode().isEmpty()) { parserResult.getMetaData().setMode(BibDatabaseModeDetection.inferMode(parserResult.getDatabase())); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java b/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java index 3fe4fb684d6..666ae01a7fb 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java @@ -283,7 +283,7 @@ private void parseAndAddEntry(String type) { LOGGER.debug("Could not parse entry", ex); parserResult.addWarning(Localization.lang("Error occurred when parsing entry") + ": '" + ex.getMessage() - + "'. " + Localization.lang("Skipped entry.")); + + "'. " + "\n\n" + Localization.lang("JabRef skipped the entry.")); } } @@ -387,8 +387,8 @@ private String purge(String context, String stringToPurge) { } // strip empty lines while ((runningIndex < indexOfAt) && - (context.charAt(runningIndex) == '\r' || - context.charAt(runningIndex) == '\n')) { + ((context.charAt(runningIndex) == '\r') || + (context.charAt(runningIndex) == '\n'))) { runningIndex++; } return context.substring(runningIndex); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java index 2ff451e6dca..afbe5af657d 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java @@ -14,6 +14,7 @@ import org.jabref.model.entry.Author; import org.jabref.model.entry.AuthorList; import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.BiblatexSoftwareField; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.StandardField; import org.jabref.model.entry.field.UnknownField; @@ -95,7 +96,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { StandardEntryType entryType = StandardEntryType.Software; // Map CFF fields to JabRef Fields - HashMap fieldMap = getFieldMappings(); + HashMap fieldMap = getFieldMappings(); for (Map.Entry property : citation.values.entrySet()) { if (fieldMap.containsKey(property.getKey())) { entryMap.put(fieldMap.get(property.getKey()), property.getValue()); @@ -120,7 +121,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { entryMap.put(StandardField.AUTHOR, authorStr); // Select DOI to keep - if (entryMap.get(StandardField.DOI) == null && citation.ids != null) { + if ((entryMap.get(StandardField.DOI) == null) && (citation.ids != null)) { List doiIds = citation.ids.stream() .filter(id -> id.type.equals("doi")) .collect(Collectors.toList()); @@ -137,14 +138,14 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { .collect(Collectors.toList()); if (swhIds.size() == 1) { - entryMap.put(StandardField.SWHID, swhIds.get(0)); + entryMap.put(BiblatexSoftwareField.SWHID, swhIds.get(0)); } else if (swhIds.size() > 1) { List relSwhIds = swhIds.stream() .filter(id -> id.split(":").length > 3) // quick filter for invalid swhids .filter(id -> id.split(":")[2].equals("rel")) .collect(Collectors.toList()); if (relSwhIds.size() == 1) { - entryMap.put(StandardField.SWHID, relSwhIds.get(0)); + entryMap.put(BiblatexSoftwareField.SWHID, relSwhIds.get(0)); } } } @@ -166,19 +167,19 @@ public boolean isRecognizedFormat(BufferedReader reader) throws IOException { try { citation = mapper.readValue(reader, CffFormat.class); - return citation != null && citation.values.get("title") != null; + return (citation != null) && (citation.values.get("title") != null); } catch (IOException e) { return false; } } - private HashMap getFieldMappings() { - HashMap fieldMappings = new HashMap<>(); + private HashMap getFieldMappings() { + HashMap fieldMappings = new HashMap<>(); fieldMappings.put("title", StandardField.TITLE); fieldMappings.put("version", StandardField.VERSION); fieldMappings.put("doi", StandardField.DOI); - fieldMappings.put("license", StandardField.LICENSE); - fieldMappings.put("repository", StandardField.REPOSITORY); + fieldMappings.put("license", BiblatexSoftwareField.LICENSE); + fieldMappings.put("repository", BiblatexSoftwareField.REPOSITORY); fieldMappings.put("url", StandardField.URL); fieldMappings.put("abstract", StandardField.ABSTRACT); fieldMappings.put("message", StandardField.COMMENT); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java index ab857becd0a..ab638ae4cf3 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java @@ -137,7 +137,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { } else if ("DT- ".equals(code)) { setOrAppend(b, new UnknownField("documenttype"), line.substring(4).trim(), ", "); } else { - setOrAppend(b, FieldFactory.parseField(code.substring(0, 2)), line.substring(4).trim(), ", "); + setOrAppend(b, FieldFactory.parseField(StandardEntryType.Book, line.substring(0, 2)), line.substring(4).trim(), ", "); } } results.add(b); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java index b502977f716..2845756b770 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java @@ -281,7 +281,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { if ("ER".equals(beg) || "EF".equals(beg) || "VR".equals(beg) || "FN".equals(beg)) { continue; } - hm.put(FieldFactory.parseField(beg), value); + hm.put(FieldFactory.parseField(type, beg), value); } } diff --git a/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java index 392fad9772b..b9e9eac39d4 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java @@ -425,7 +425,7 @@ private void addArticleIdList(Map fields, ArticleIdList articleId if ("pubmed".equals(id.getIdType())) { fields.put(StandardField.PMID, id.getContent()); } else { - fields.put(FieldFactory.parseField(id.getIdType()), id.getContent()); + fields.put(FieldFactory.parseField(StandardEntryType.Article, id.getIdType()), id.getContent()); } } } @@ -499,7 +499,7 @@ private void addKeyWords(Map fields, List allKeyword private void addOtherId(Map fields, List otherID) { for (OtherID id : otherID) { if ((id.getSource() != null) && (id.getContent() != null)) { - fields.put(FieldFactory.parseField(id.getSource()), id.getContent()); + fields.put(FieldFactory.parseField(StandardEntryType.Article, id.getSource()), id.getContent()); } } } diff --git a/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java b/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java index eb1e905adca..5f3d5e29c52 100644 --- a/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java +++ b/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java @@ -65,6 +65,7 @@ public MetaData parse(MetaData metaData, Map data, Character key String user = entry.getKey().substring(MetaData.FILE_DIRECTORY.length() + 1); metaData.setUserFileDirectory(user, getSingleItem(value)); } else if (entry.getKey().startsWith(MetaData.SELECTOR_META_PREFIX)) { + // edge case, it might be one special field e.g. article from biblatex-apa, but we can't distinguish this from any other field and rather prefer to handle it as UnknownField metaData.addContentSelector(ContentSelectors.parse(FieldFactory.parseField(entry.getKey().substring(MetaData.SELECTOR_META_PREFIX.length())), StringUtil.unquote(entry.getValue(), MetaData.ESCAPE_CHARACTER))); } else if (entry.getKey().startsWith(MetaData.FILE_DIRECTORY + "Latex-")) { // The user name comes directly after "FILE_DIRECTORYLatex-" diff --git a/src/main/java/org/jabref/logic/layout/format/NameFormatter.java b/src/main/java/org/jabref/logic/layout/format/NameFormatter.java index 919ff4df2d5..e8812235435 100644 --- a/src/main/java/org/jabref/logic/layout/format/NameFormatter.java +++ b/src/main/java/org/jabref/logic/layout/format/NameFormatter.java @@ -5,7 +5,7 @@ import java.util.Map; import java.util.Objects; -import org.jabref.logic.bst.BibtexNameFormatter; +import org.jabref.logic.bst.util.BstNameFormatter; import org.jabref.logic.layout.LayoutFormatter; import org.jabref.model.entry.AuthorList; @@ -86,7 +86,7 @@ private static String format(String toFormat, AuthorList al, String[] formats) { for (int i = 1; i <= al.getNumberOfAuthors(); i++) { for (int j = 1; j < formats.length; j += 2) { if ("*".equals(formats[j])) { - sb.append(BibtexNameFormatter.formatName(toFormat, i, formats[j + 1], null)); + sb.append(BstNameFormatter.formatName(toFormat, i, formats[j + 1])); break; } else { String[] range = formats[j].split("\\.\\."); @@ -112,7 +112,7 @@ private static String format(String toFormat, AuthorList al, String[] formats) { } if ((s <= i) && (i <= e)) { - sb.append(BibtexNameFormatter.formatName(toFormat, i, formats[j + 1], null)); + sb.append(BstNameFormatter.formatName(toFormat, i, formats[j + 1])); break; } } diff --git a/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java b/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java index 24e897ae926..8ebc6bddd04 100644 --- a/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java +++ b/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java @@ -2,6 +2,7 @@ import java.util.List; import java.util.Queue; +import java.util.Set; import java.util.concurrent.ConcurrentLinkedQueue; import org.jabref.gui.util.BackgroundTask; @@ -88,12 +89,17 @@ public void createIndex(PdfIndexer indexer) { enqueueTask(() -> indexer.createIndex()); } - public void addToIndex(PdfIndexer indexer, BibDatabaseContext databaseContext) { + public void updateIndex(PdfIndexer indexer, BibDatabaseContext databaseContext) { + Set pathsToRemove = indexer.getListOfFilePaths(); for (BibEntry entry : databaseContext.getEntries()) { for (LinkedFile file : entry.getFiles()) { enqueueTask(() -> indexer.addToIndex(entry, file, databaseContext)); + pathsToRemove.remove(file.getLink()); } } + for (String pathToRemove : pathsToRemove) { + enqueueTask(() -> indexer.removeFromIndex(pathToRemove)); + } } public void addToIndex(PdfIndexer indexer, BibEntry entry, BibDatabaseContext databaseContext) { @@ -108,7 +114,7 @@ public void addToIndex(PdfIndexer indexer, BibEntry entry, List link public void removeFromIndex(PdfIndexer indexer, BibEntry entry, List linkedFiles) { for (LinkedFile file : linkedFiles) { - enqueueTask(() -> indexer.removeFromIndex(entry, file)); + enqueueTask(() -> indexer.removeFromIndex(file.getLink())); } } diff --git a/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java b/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java index 3b91be44f71..975cf83b1ec 100644 --- a/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java +++ b/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java @@ -4,8 +4,10 @@ import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.attribute.BasicFileAttributes; +import java.util.HashSet; import java.util.List; import java.util.Optional; +import java.util.Set; import java.util.concurrent.TimeUnit; import org.jabref.gui.LibraryTab; @@ -25,6 +27,8 @@ import org.apache.lucene.index.IndexWriterConfig; import org.apache.lucene.index.Term; import org.apache.lucene.search.IndexSearcher; +import org.apache.lucene.search.MatchAllDocsQuery; +import org.apache.lucene.search.ScoreDoc; import org.apache.lucene.search.TermQuery; import org.apache.lucene.search.TopDocs; import org.apache.lucene.store.Directory; @@ -110,19 +114,16 @@ public void addToIndex(BibEntry entry, LinkedFile linkedFile, BibDatabaseContext } /** - * Removes a pdf file linked to one entry in the database from the index + * Removes a pdf file identified by its path from the index * - * @param entry the entry the file is linked to - * @param linkedFile the link to the file to be removed + * @param linkedFilePath the path to the file to be removed */ - public void removeFromIndex(BibEntry entry, LinkedFile linkedFile) { + public void removeFromIndex(String linkedFilePath) { try (IndexWriter indexWriter = new IndexWriter( directoryToIndex, new IndexWriterConfig( new EnglishStemAnalyzer()).setOpenMode(IndexWriterConfig.OpenMode.CREATE_OR_APPEND))) { - if (!entry.getFiles().isEmpty()) { - indexWriter.deleteDocuments(new Term(SearchFieldConstants.PATH, linkedFile.getLink())); - } + indexWriter.deleteDocuments(new Term(SearchFieldConstants.PATH, linkedFilePath)); indexWriter.commit(); } catch (IOException e) { LOGGER.warn("Could not initialize the IndexWriter!", e); @@ -145,7 +146,7 @@ public void removeFromIndex(BibEntry entry) { */ public void removeFromIndex(BibEntry entry, List linkedFiles) { for (LinkedFile linkedFile : linkedFiles) { - removeFromIndex(entry, linkedFile); + removeFromIndex(linkedFile.getLink()); } } @@ -224,4 +225,25 @@ private void writeToIndex(BibEntry entry, LinkedFile linkedFile) { LOGGER.warn("Could not add the document {} to the index!", linkedFile.getLink(), e); } } + + /** + * Lists the paths of all the files that are stored in the index + * + * @return all file paths + */ + public Set getListOfFilePaths() { + Set paths = new HashSet<>(); + try (IndexReader reader = DirectoryReader.open(directoryToIndex)) { + IndexSearcher searcher = new IndexSearcher(reader); + MatchAllDocsQuery query = new MatchAllDocsQuery(); + TopDocs allDocs = searcher.search(query, Integer.MAX_VALUE); + for (ScoreDoc scoreDoc : allDocs.scoreDocs) { + Document doc = reader.document(scoreDoc.doc); + paths.add(doc.getField(SearchFieldConstants.PATH).stringValue()); + } + } catch (IOException e) { + return paths; + } + return paths; + } } diff --git a/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java b/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java index 260ec297c24..4b1a14fe1c9 100644 --- a/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java +++ b/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java @@ -25,6 +25,8 @@ public class SharedDatabasePreferences { private static final String SHARED_DATABASE_NAME = "sharedDatabaseName"; private static final String SHARED_DATABASE_USER = "sharedDatabaseUser"; private static final String SHARED_DATABASE_PASSWORD = "sharedDatabasePassword"; + private static final String SHARED_DATABASE_FOLDER = "sharedDatabaseFolder"; + private static final String SHARED_DATABASE_AUTOSAVE = "sharedDatabaseAutosave"; private static final String SHARED_DATABASE_REMEMBER_PASSWORD = "sharedDatabaseRememberPassword"; private static final String SHARED_DATABASE_USE_SSL = "sharedDatabaseUseSSL"; private static final String SHARED_DATABASE_KEYSTORE_FILE = "sharedDatabaseKeyStoreFile"; @@ -77,6 +79,14 @@ public boolean getRememberPassword() { return internalPrefs.getBoolean(SHARED_DATABASE_REMEMBER_PASSWORD, false); } + public Optional getFolder() { + return getOptionalValue(SHARED_DATABASE_FOLDER); + } + + public boolean getAutosave() { + return internalPrefs.getBoolean(SHARED_DATABASE_AUTOSAVE, false); + } + public boolean isUseSSL() { return internalPrefs.getBoolean(SHARED_DATABASE_USE_SSL, false); } @@ -109,6 +119,14 @@ public void setRememberPassword(boolean rememberPassword) { internalPrefs.putBoolean(SHARED_DATABASE_REMEMBER_PASSWORD, rememberPassword); } + public void setFolder(String folder) { + internalPrefs.put(SHARED_DATABASE_FOLDER, folder); + } + + public void setAutosave(boolean autosave) { + internalPrefs.putBoolean(SHARED_DATABASE_AUTOSAVE, autosave); + } + public void setUseSSL(boolean useSSL) { internalPrefs.putBoolean(SHARED_DATABASE_USE_SSL, useSSL); } diff --git a/src/main/java/org/jabref/model/database/BibDatabase.java b/src/main/java/org/jabref/model/database/BibDatabase.java index 72a559ff663..52e4e841ce9 100644 --- a/src/main/java/org/jabref/model/database/BibDatabase.java +++ b/src/main/java/org/jabref/model/database/BibDatabase.java @@ -20,6 +20,7 @@ import javafx.collections.FXCollections; import javafx.collections.ObservableList; +import javafx.collections.ObservableSet; import org.jabref.logic.bibtex.FieldWriter; import org.jabref.model.database.event.EntriesAddedEvent; @@ -29,6 +30,7 @@ import org.jabref.model.entry.Month; import org.jabref.model.entry.event.EntriesEventSource; import org.jabref.model.entry.event.EntryChangedEvent; +import org.jabref.model.entry.event.FieldAddedOrRemovedEvent; import org.jabref.model.entry.event.FieldChangedEvent; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.FieldFactory; @@ -52,6 +54,8 @@ public class BibDatabase { * State attributes */ private final ObservableList entries = FXCollections.synchronizedObservableList(FXCollections.observableArrayList(BibEntry::getObservables)); + + private final ObservableSet visibleFields = FXCollections.observableSet(); private Map bibtexStrings = new ConcurrentHashMap<>(); private final EventBus eventBus = new EventBus(); @@ -136,13 +140,8 @@ public ObservableList getEntries() { * * @return set of fieldnames, that are visible */ - public Set getAllVisibleFields() { - Set allFields = new TreeSet<>(Comparator.comparing(Field::getName)); - for (BibEntry e : getEntries()) { - allFields.addAll(e.getFields()); - } - return allFields.stream().filter(field -> !FieldFactory.isInternalField(field)) - .collect(Collectors.toSet()); + public ObservableSet getAllVisibleFields() { + return visibleFields; } /** @@ -214,6 +213,8 @@ public synchronized void insertEntries(List newEntries, EntriesEventSo eventBus.post(new EntriesAddedEvent(newEntries, newEntries.get(0), eventSource)); } entries.addAll(newEntries); + + updateVisibleFields(); } public synchronized void removeEntry(BibEntry bibEntry) { @@ -251,6 +252,7 @@ public synchronized void removeEntries(List toBeDeleted, EntriesEventS boolean anyRemoved = entries.removeIf(entry -> ids.contains(entry.getId())); if (anyRemoved) { eventBus.post(new EntriesRemovedEvent(toBeDeleted, eventSource)); + updateVisibleFields(); } } @@ -584,6 +586,30 @@ private void relayEntryChangeEvent(FieldChangedEvent event) { eventBus.post(event); } + @Subscribe + private void listen(FieldAddedOrRemovedEvent event) { + // When a field is removed from an entry we can't tell if it's + // still present in other entries, and thus we can't remove it + // from the set of visible fields. However, when a new field is added + // to any entry, we can simply add it to the set because we're + // going to add it whether other entries have it or not + boolean isAdded = visibleFields.add(event.getField()); + if (!isAdded) { + updateVisibleFields(); + } + } + + private void updateVisibleFields() { + visibleFields.clear(); + Set allFields = new TreeSet<>(Comparator.comparing(Field::getName)); + for (BibEntry e : getEntries()) { + allFields.addAll(e.getFields()); + } + visibleFields.addAll(allFields.stream().filter(field -> !FieldFactory.isInternalField(field)) + .filter(field -> StringUtil.isNotBlank(field.getName())) + .collect(Collectors.toSet())); + } + public Optional getReferencedEntry(BibEntry entry) { return entry.getField(StandardField.CROSSREF).flatMap(this::getEntryByCitationKey); } diff --git a/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java b/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java index b737596abd6..836cd39dac8 100644 --- a/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java +++ b/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java @@ -11,6 +11,7 @@ import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.field.BibField; import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.types.BiblatexAPAEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexSoftwareEntryTypeDefinitions; import org.jabref.model.entry.types.BibtexEntryTypeDefinitions; @@ -21,7 +22,7 @@ public class BibEntryTypesManager { public static final String ENTRYTYPE_FLAG = "jabref-entrytype: "; private final InternalEntryTypes BIBTEX = new InternalEntryTypes(Stream.concat(BibtexEntryTypeDefinitions.ALL.stream(), IEEETranEntryTypeDefinitions.ALL.stream()).collect(Collectors.toList())); - private final InternalEntryTypes BIBLATEX = new InternalEntryTypes(Stream.concat(BiblatexEntryTypeDefinitions.ALL.stream(), BiblatexSoftwareEntryTypeDefinitions.ALL.stream()).collect(Collectors.toList())); + private final InternalEntryTypes BIBLATEX = new InternalEntryTypes(Stream.concat(BiblatexEntryTypeDefinitions.ALL.stream(), Stream.concat(BiblatexSoftwareEntryTypeDefinitions.ALL.stream(), BiblatexAPAEntryTypeDefinitions.ALL.stream())).collect(Collectors.toList())); public BibEntryTypesManager() { } @@ -99,6 +100,7 @@ public List getAllCustomTypes(BibDatabaseMode mode) { return customizedTypes.stream() .filter(entryType -> BiblatexEntryTypeDefinitions.ALL.stream().noneMatch(biblatexType -> biblatexType.getType().equals(entryType.getType()))) .filter(entryType -> BiblatexSoftwareEntryTypeDefinitions.ALL.stream().noneMatch(biblatexSoftware -> biblatexSoftware.getType().equals(entryType.getType()))) + .filter(entryType -> BiblatexAPAEntryTypeDefinitions.ALL.stream().noneMatch(biblatexAPA -> biblatexAPA.getType().equals(entryType.getType()))) .collect(Collectors.toList()); } } diff --git a/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java b/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java new file mode 100644 index 00000000000..b2938963efe --- /dev/null +++ b/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java @@ -0,0 +1,82 @@ +package org.jabref.model.entry.field; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; +import java.util.Optional; +import java.util.Set; + +import org.jabref.model.entry.types.BiblatexApaEntryType; + +public enum BiblatexApaField implements Field { + + AMENDMENT("amendment"), + ARTICLE("article"), + CITATION("citation"), + CITATION_CITEORG("citation_citeorg"), + CITATION_CITEDATE("citation_citedate", FieldProperty.DATE), + CITATION_CITEINFO("citation_citeinfo"), + SECTION("section", FieldProperty.NUMERIC), + SOURCE("source"); + + private final String name; + private final String displayName; + private final Set properties; + + BiblatexApaField(String name) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexApaField(String name, String displayName) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexApaField(String name, String displayName, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.of(first, rest); + } + + BiblatexApaField(String name, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.of(first, rest); + } + + public static Optional fromName(T type, String name) { + if (!(type instanceof BiblatexApaEntryType)) { + return Optional.empty(); + } + return Arrays.stream(BiblatexApaField.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public Set getProperties() { + return Collections.unmodifiableSet(properties); + } + + @Override + public String getName() { + return name; + } + + @Override + public boolean isStandardField() { + return false; + } + + @Override + public String getDisplayName() { + if (displayName == null) { + return Field.super.getDisplayName(); + } else { + return displayName; + } + } +} diff --git a/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java b/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java new file mode 100644 index 00000000000..fc3bb4dcc48 --- /dev/null +++ b/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java @@ -0,0 +1,82 @@ +package org.jabref.model.entry.field; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; +import java.util.Optional; +import java.util.Set; + +import org.jabref.model.entry.types.BiblatexSoftwareEntryType; + +public enum BiblatexSoftwareField implements Field { + + HALID("hal_id"), + HALVERSION("hal_version"), + INTRODUCEDIN("introducedin"), + LICENSE("license"), + RELATEDTYPE("relatedtype"), + RELATEDSTRING("relatedstring"), + REPOSITORY("repository"), + SWHID("swhid"); + + private final String name; + private final String displayName; + private final Set properties; + + BiblatexSoftwareField(String name) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexSoftwareField(String name, String displayName) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexSoftwareField(String name, String displayName, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.of(first, rest); + } + + BiblatexSoftwareField(String name, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.of(first, rest); + } + + public static Optional fromName(T type, String name) { + if (!(type instanceof BiblatexSoftwareEntryType)) { + return Optional.empty(); + } + return Arrays.stream(BiblatexSoftwareField.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public Set getProperties() { + return Collections.unmodifiableSet(properties); + } + + @Override + public String getName() { + return name; + } + + @Override + public boolean isStandardField() { + return false; + } + + @Override + public String getDisplayName() { + if (displayName == null) { + return Field.super.getDisplayName(); + } else { + return displayName; + } + } +} diff --git a/src/main/java/org/jabref/model/entry/field/FieldFactory.java b/src/main/java/org/jabref/model/entry/field/FieldFactory.java index b1c2f3b99b9..07a34b039d3 100644 --- a/src/main/java/org/jabref/model/entry/field/FieldFactory.java +++ b/src/main/java/org/jabref/model/entry/field/FieldFactory.java @@ -73,13 +73,23 @@ public static String serializeFieldsList(Collection fields) { .collect(Collectors.joining(DELIMITER)); } + public static Field parseField(T type, String fieldName) { + return OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + InternalField.fromName(fieldName), + StandardField.fromName(fieldName)), + SpecialField.fromName(fieldName)), + IEEEField.fromName(fieldName)), + BiblatexSoftwareField.fromName(type, fieldName)), + BiblatexApaField.fromName(type, fieldName)) + .orElse(new UnknownField(fieldName)); + } + public static Field parseField(String fieldName) { - return OptionalUtil.orElse(OptionalUtil.orElse(OptionalUtil.orElse( - InternalField.fromName(fieldName), - StandardField.fromName(fieldName)), - SpecialField.fromName(fieldName)), - IEEEField.fromName(fieldName)) - .orElse(new UnknownField(fieldName)); + return parseField(null, fieldName); } public static Set getKeyFields() { @@ -138,6 +148,8 @@ private static Set getFieldsFiltered(Predicate selector) { private static Set getAllFields() { Set fields = new HashSet<>(); + fields.addAll(EnumSet.allOf(BiblatexApaField.class)); + fields.addAll(EnumSet.allOf(BiblatexSoftwareField.class)); fields.addAll(EnumSet.allOf(IEEEField.class)); fields.addAll(EnumSet.allOf(InternalField.class)); fields.addAll(EnumSet.allOf(SpecialField.class)); diff --git a/src/main/java/org/jabref/model/entry/field/IEEEField.java b/src/main/java/org/jabref/model/entry/field/IEEEField.java index d00bec02eee..f56ce79ea99 100644 --- a/src/main/java/org/jabref/model/entry/field/IEEEField.java +++ b/src/main/java/org/jabref/model/entry/field/IEEEField.java @@ -31,7 +31,7 @@ public enum IEEEField implements Field { this.properties = EnumSet.of(first, rest); } - public static Optional fromName(String name) { + public static Optional fromName(String name) { return Arrays.stream(IEEEField.values()) .filter(field -> field.getName().equalsIgnoreCase(name)) .findAny(); diff --git a/src/main/java/org/jabref/model/entry/field/StandardField.java b/src/main/java/org/jabref/model/entry/field/StandardField.java index ef3646802e0..bcf9b710b7d 100644 --- a/src/main/java/org/jabref/model/entry/field/StandardField.java +++ b/src/main/java/org/jabref/model/entry/field/StandardField.java @@ -59,14 +59,11 @@ public enum StandardField implements Field { FOREWORD("foreword", FieldProperty.PERSON_NAMES), FOLDER("folder"), GENDER("gender", FieldProperty.GENDER), - HALID("hal_id"), - HALVERSION("hal_version"), HOLDER("holder", FieldProperty.PERSON_NAMES), HOWPUBLISHED("howpublished"), IDS("ids", FieldProperty.MULTIPLE_ENTRY_LINK), INSTITUTION("institution"), INTRODUCTION("introduction", FieldProperty.PERSON_NAMES), - INTRODUCEDIN("introducedin"), ISBN("isbn", "ISBN", FieldProperty.ISBN), ISRN("isrn", "ISRN"), ISSN("issn", "ISSN"), @@ -81,7 +78,6 @@ public enum StandardField implements Field { LANGUAGE("language", FieldProperty.LANGUAGE), LABEL("label"), LIBRARY("library"), - LICENSE("license"), LOCATION("location"), MAINSUBTITLE("mainsubtitle", FieldProperty.BOOK_NAME), MAINTITLE("maintitle", FieldProperty.BOOK_NAME), @@ -106,10 +102,7 @@ public enum StandardField implements Field { PUBSTATE("pubstate", FieldProperty.PUBLICATION_STATE), PRIMARYCLASS("primaryclass"), RELATED("related", FieldProperty.MULTIPLE_ENTRY_LINK), - RELATEDTYPE("relatedtype"), - RELATEDSTRING("relatedstring"), REPORTNO("reportno"), - REPOSITORY("repository"), REVIEW("review"), REVISION("revision"), SCHOOL("school"), @@ -120,7 +113,6 @@ public enum StandardField implements Field { SORTKEY("sortkey"), SORTNAME("sortname", FieldProperty.PERSON_NAMES), SUBTITLE("subtitle"), - SWHID("swhid"), TITLE("title"), TITLEADDON("titleaddon"), TRANSLATOR("translator", FieldProperty.PERSON_NAMES), diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java b/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java new file mode 100644 index 00000000000..f5c0b89b0be --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java @@ -0,0 +1,43 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.List; + +import org.jabref.model.entry.BibEntryType; +import org.jabref.model.entry.BibEntryTypeBuilder; +import org.jabref.model.entry.field.BiblatexApaField; +import org.jabref.model.entry.field.StandardField; + +public class BiblatexAPAEntryTypeDefinitions { + + private static final BibEntryType JURISDICTION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Jurisdiction) + .withImportantFields(StandardField.ORGANIZATION, BiblatexApaField.CITATION_CITEORG, BiblatexApaField.CITATION_CITEDATE, BiblatexApaField.CITATION_CITEDATE, StandardField.ORIGDATE) + .withRequiredFields(StandardField.TITLE, BiblatexApaField.CITATION, BiblatexApaField.CITATION_CITEINFO, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType LEGISLATION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legislation) + .withImportantFields(StandardField.TITLEADDON, StandardField.ORIGDATE) + .withRequiredFields(StandardField.TITLE, StandardField.LOCATION, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType LEGADMINMATERIAL = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legadminmaterial) + .withImportantFields(StandardField.NUMBER, StandardField.SHORTTITLE, StandardField.NOTE, StandardField.KEYWORDS) + .withRequiredFields(StandardField.TITLE, BiblatexApaField.CITATION, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType CONSTITUTION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Constitution) + .withImportantFields(BiblatexApaField.ARTICLE, BiblatexApaField.AMENDMENT, StandardField.EVENTDATE, StandardField.KEYWORDS, StandardField.PART, BiblatexApaField.SECTION) + .withRequiredFields(BiblatexApaField.SOURCE, StandardField.TYPE) + .build(); + + private static final BibEntryType LEGAL = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legal) + .withRequiredFields(StandardField.TITLE, StandardField.DATE, StandardField.URI, StandardField.KEYWORDS, StandardField.PART, BiblatexApaField.SECTION) + .build(); + + public static final List ALL = Arrays.asList(JURISDICTION, LEGISLATION, LEGADMINMATERIAL, CONSTITUTION, LEGAL); +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java b/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java new file mode 100644 index 00000000000..6a1d8d3abe2 --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java @@ -0,0 +1,36 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.Locale; +import java.util.Optional; + +public enum BiblatexApaEntryType implements EntryType { + + Legislation("Legislation"), + Legadminmaterial("Legadminmaterial"), + Jurisdiction("Jurisdiction"), + Constitution("Constitution"), + Legal("Legal"); + + private final String displayName; + + BiblatexApaEntryType(String displayName) { + this.displayName = displayName; + } + + @Override + public String getName() { + return displayName.toLowerCase(Locale.ENGLISH); + } + + @Override + public String getDisplayName() { + return displayName; + } + + public static Optional fromName(String name) { + return Arrays.stream(BiblatexApaEntryType.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java new file mode 100644 index 00000000000..041c48e592a --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java @@ -0,0 +1,35 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.Locale; +import java.util.Optional; + +public enum BiblatexSoftwareEntryType implements EntryType { + + Dataset("Dataset"), + SoftwareVersion("SoftwareVersion"), + SoftwareModule("SoftwareModule"), + CodeFragment("CodeFragment"); + + private final String displayName; + + BiblatexSoftwareEntryType(String displayName) { + this.displayName = displayName; + } + + public static Optional fromName(String name) { + return Arrays.stream(BiblatexSoftwareEntryType.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public String getName() { + return displayName.toLowerCase(Locale.ENGLISH); + } + + @Override + public String getDisplayName() { + return displayName; + } +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java index 0fc48e6d904..6a2e7f0851e 100644 --- a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java +++ b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java @@ -5,6 +5,7 @@ import org.jabref.model.entry.BibEntryType; import org.jabref.model.entry.BibEntryTypeBuilder; +import org.jabref.model.entry.field.BiblatexSoftwareField; import org.jabref.model.entry.field.OrFields; import org.jabref.model.entry.field.StandardField; @@ -12,40 +13,40 @@ public class BiblatexSoftwareEntryTypeDefinitions { private static final BibEntryType SOFTWARE = new BibEntryTypeBuilder() .withType(StandardEntryType.Software) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.VERSION, StandardField.YEAR) .build(); private static final BibEntryType SOFTWAREVERSION = new BibEntryTypeBuilder() - .withType(StandardEntryType.SoftwareVersion) - .withImportantFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, StandardField.HALID, StandardField.HALVERSION, - StandardField.INSTITUTION, StandardField.INTRODUCEDIN, StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, - StandardField.PUBLISHER, StandardField.RELATED, StandardField.RELATEDTYPE, StandardField.RELATEDSTRING, - StandardField.REPOSITORY, StandardField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) + .withType(BiblatexSoftwareEntryType.SoftwareVersion) + .withImportantFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, + StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, + StandardField.PUBLISHER, StandardField.RELATED, BiblatexSoftwareField.RELATEDTYPE, BiblatexSoftwareField.RELATEDSTRING, + BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.YEAR, StandardField.VERSION) - .withDetailFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, StandardField.HALID, StandardField.HALVERSION, - StandardField.INSTITUTION, StandardField.INTRODUCEDIN, StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, - StandardField.PUBLISHER, StandardField.RELATED, StandardField.RELATEDTYPE, StandardField.RELATEDSTRING, - StandardField.REPOSITORY, StandardField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) + .withDetailFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, + StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, + StandardField.PUBLISHER, StandardField.RELATED, BiblatexSoftwareField.RELATEDTYPE, BiblatexSoftwareField.RELATEDSTRING, + BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.YEAR) .build(); private static final BibEntryType SOFTWAREMODULE = new BibEntryTypeBuilder() - .withType(StandardEntryType.SoftwareModule) + .withType(BiblatexSoftwareEntryType.SoftwareModule) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(StandardField.AUTHOR, StandardField.SUBTITLE, StandardField.URL, StandardField.YEAR) .build(); private static final BibEntryType CODEFRAGMENT = new BibEntryTypeBuilder() - .withType(StandardEntryType.CodeFragment) + .withType(BiblatexSoftwareEntryType.CodeFragment) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(StandardField.URL) .build(); diff --git a/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java b/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java index 1c94f6dd9a0..6b40a56c4f4 100644 --- a/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java +++ b/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java @@ -49,6 +49,8 @@ private static boolean isBiblatex(EntryType type) { public static EntryType parse(String typeName) { List types = new ArrayList<>(Arrays.asList(StandardEntryType.values())); types.addAll(Arrays.asList(IEEETranEntryType.values())); + types.addAll(Arrays.asList(BiblatexSoftwareEntryType.values())); + types.addAll(Arrays.asList(BiblatexApaEntryType.values())); types.addAll(Arrays.asList(SystematicLiteratureReviewStudyEntryType.values())); return types.stream().filter(type -> type.getName().equals(typeName.toLowerCase(Locale.ENGLISH))).findFirst().orElse(new UnknownEntryType(typeName)); diff --git a/src/main/java/org/jabref/model/entry/types/StandardEntryType.java b/src/main/java/org/jabref/model/entry/types/StandardEntryType.java index fb2922e61a0..6f6f91ac6a5 100644 --- a/src/main/java/org/jabref/model/entry/types/StandardEntryType.java +++ b/src/main/java/org/jabref/model/entry/types/StandardEntryType.java @@ -36,10 +36,7 @@ public enum StandardEntryType implements EntryType { Thesis("Thesis"), WWW("WWW"), Software("Software"), - Dataset("Dataset"), - SoftwareVersion("SoftwareVersion"), - SoftwareModule("SoftwareModule"), - CodeFragment("CodeFragment"); + Dataset("Dataset"); private final String displayName; diff --git a/src/main/java/org/jabref/model/metadata/MetaData.java b/src/main/java/org/jabref/model/metadata/MetaData.java index 238658803d3..9b0ed2ae856 100644 --- a/src/main/java/org/jabref/model/metadata/MetaData.java +++ b/src/main/java/org/jabref/model/metadata/MetaData.java @@ -68,6 +68,7 @@ public class MetaData { private final ContentSelectors contentSelectors = new ContentSelectors(); private final Map> unknownMetaData = new HashMap<>(); private boolean isEventPropagationEnabled = true; + private boolean encodingExplicitlySupplied; /** * Constructs an empty metadata. @@ -291,6 +292,17 @@ public void setEncoding(Charset encoding, ChangePropagation postChanges) { } } + public boolean getEncodingExplicitlySupplied() { + return encodingExplicitlySupplied; + } + + /** + * Sets the indication whether the encoding was set using "% Encoding: ..." or whether it was detected "magically" + */ + public void setEncodingExplicitlySupplied(boolean encodingExplicitlySupplied) { + this.encodingExplicitlySupplied = encodingExplicitlySupplied; + } + /** * If disabled {@link MetaDataChangedEvent} will not be posted. */ @@ -349,6 +361,7 @@ public boolean equals(Object o) { return (isProtected == metaData.isProtected) && Objects.equals(groupsRoot.getValue(), metaData.groupsRoot.getValue()) && Objects.equals(encoding, metaData.encoding) + && Objects.equals(encodingExplicitlySupplied, metaData.encodingExplicitlySupplied) && Objects.equals(saveOrderConfig, metaData.saveOrderConfig) && Objects.equals(citeKeyPatterns, metaData.citeKeyPatterns) && Objects.equals(userFileDirectory, metaData.userFileDirectory) @@ -362,7 +375,7 @@ public boolean equals(Object o) { @Override public int hashCode() { - return Objects.hash(groupsRoot.getValue(), encoding, saveOrderConfig, citeKeyPatterns, userFileDirectory, + return Objects.hash(groupsRoot.getValue(), encoding, encodingExplicitlySupplied, saveOrderConfig, citeKeyPatterns, userFileDirectory, defaultCiteKeyPattern, saveActions, mode, isProtected, defaultFileDirectory); } } diff --git a/src/main/java/org/jabref/model/strings/StringUtil.java b/src/main/java/org/jabref/model/strings/StringUtil.java index 19f0b7a4542..2a443306e4d 100644 --- a/src/main/java/org/jabref/model/strings/StringUtil.java +++ b/src/main/java/org/jabref/model/strings/StringUtil.java @@ -750,4 +750,15 @@ public static String quoteStringIfSpaceIsContained(String string) { return string; } } + + /** + * Checks if the given string contains any whitespace characters. The supported whitespace characters + * are the set of characters matched by {@code \s} in regular expressions, which are {@code [ \t\n\x0B\f\r]}. + * + * @param s The string to check + * @return {@code True} if the given string does contain at least one whitespace character, {@code False} otherwise + * */ + public static boolean containsWhitespace(String s) { + return s.chars().anyMatch(Character::isWhitespace); + } } diff --git a/src/main/resources/l10n/JabRef_da.properties b/src/main/resources/l10n/JabRef_da.properties index c07cb809a45..97eeb562207 100644 --- a/src/main/resources/l10n/JabRef_da.properties +++ b/src/main/resources/l10n/JabRef_da.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 matcher regulærudtrykket %1 %0\ matches\ the\ term\ %1=%0 matcher udtrykket %1 @@ -122,8 +123,6 @@ Cite\ command=Citations-kommando Clear=Ryd -Clear\ fields=Ryd felter - Close\ dialog=Luk dialog @@ -574,11 +573,18 @@ Remove\ entry\ from\ import=Fjern post fra import Remove\ group=Fjern gruppe -Remove\ group\ "%0"?=Fjern gruppen "%0"? +Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Fjern gruppen "%0" og dens undergrupper? -Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernede gruppen "%0" og dens undergrupper + + Remove\ link=Slet link @@ -587,8 +593,6 @@ Remove\ old\ entry=Fjern gammel post Removed\ group\ "%0".=Fjernede gruppen "%0" -Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernede gruppen "%0" og dens undergrupper - Removed\ string=Streng fjernet Renamed\ string=ændrede navn på streng @@ -642,9 +646,6 @@ Select\ file\ from\ ZIP-archive=Vælg fil fra ZIP-fil Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Vælg forgreningerne for at inspicere og acceptere eller forkaste ændringer -Set\ field=Sæt felt -Set\ fields=Sæt felter - Settings=Indstillinger Shortcut=Genvej @@ -675,7 +676,6 @@ Size=Størrelse Skipped\ -\ No\ PDF\ linked=Sprang over - ingen PDF-fil linket Skipped\ -\ PDF\ does\ not\ exist=Sprang over - PDF-filen findes ikke -Skipped\ entry.=Sprang over post. source\ edit=redigering af kilde @@ -787,8 +787,6 @@ MIME\ type=MIME-type This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ running\ instance\ of\ JabRef\ instead\ of\ opening\ a\ new\ instance.\ For\ instance,\ this\ is\ useful\ when\ you\ open\ a\ file\ in\ JabRef\ from\ your\ web\ browser.\ Note\ that\ this\ will\ prevent\ you\ from\ running\ more\ than\ one\ instance\ of\ JabRef\ at\ a\ time.=Denne funktion tillader, at flere filer kan åbnes eller importeres i en allerede kørende JabRef i stedet for at åbne programmet påny. For eksempel er dette praktisk, når du åbner filer i JabRef fra din web browser. Bemærk at dette vil forhindre dig i at køre mere end en instans af JabRef ad gangen. Rename\ field=Omdøb felt -Rename\ field\ to=Omdøb felt til -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flyt indhold af et felt til et felt med et andet navn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan ikke bruge port %0 til fjernstyring; et andet program bruger den måske. Prøv en anden port. @@ -914,6 +912,7 @@ Finished\ automatically\ setting\ external\ links.=Fuldførte automatisk udfyldn + Open\ folder=Åben mappe @@ -1088,6 +1087,8 @@ Default\ pattern=Standardmønster + + diff --git a/src/main/resources/l10n/JabRef_de.properties b/src/main/resources/l10n/JabRef_de.properties index d9f4e1272f6..ee962e6a73b 100644 --- a/src/main/resources/l10n/JabRef_de.properties +++ b/src/main/resources/l10n/JabRef_de.properties @@ -19,7 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 Einträge -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 Export erfolgreich. Ordner mit der gespeicherten Datei öffnen? +Export\ operation\ finished\ successfully.=Exportvorgang erfolgreich abgeschlossen. + +Reveal\ in\ File\ Explorer=Im Datei-Explorer anzeigen %0\ matches\ the\ regular\ expression\ %1=%0 exakt dem regulären Ausdruck %1 entspricht @@ -151,8 +153,6 @@ Cite\ command=Cite-Befehl Clear=Zurücksetzen -Clear\ fields=Felder löschen - Open\ /\ close\ entry\ editor=Eintragseditor öffnen / schließen Close\ dialog=Dialog schließen @@ -703,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Ausgewählte Einträge aus dieser G Remove\ group=Gruppe löschen -Remove\ group\ "%0"?=Gruppe "%0" löschen? +Remove\ group\ and\ subgroups=Gruppe und Untergruppen löschen + +Remove\ groups\ and\ subgroups=Gruppen und Untergruppen entfernen + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Alle ausgewählten Gruppen entfernen und ihre Untergruppen beibehalten? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Gruppe "%0" entfernen und Untergruppen beibehalten? + +Remove\ groups=Gruppen entfernen + +Removed\ all\ selected\ groups.=Alle ausgewählten Gruppen entfernen. Remove\ group\ "%0"\ and\ its\ subgroups?=Gruppe "%0" inklusive Untergruppen löschen? -Remove\ group\ and\ subgroups=Gruppe und Untergruppen löschen +Removed\ group\ "%0"\ and\ its\ subgroups.=Gruppe "%0" inklusive Untergruppen gelöscht. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Alle ausgewählten Gruppen und ihre Untergruppen entfernen? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Alle ausgewählten Gruppen und ihre Untergruppen entfernt. Remove\ link=Link löschen @@ -717,8 +731,6 @@ Remove\ string\ %0=Entferne String %0 Removed\ group\ "%0".=Gruppe "%0" gelöscht. -Removed\ group\ "%0"\ and\ its\ subgroups.=Gruppe "%0" inklusive Untergruppen gelöscht. - Removed\ string=String gelöscht Renamed\ string=String umbenannt @@ -790,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Eintrag aus der ZIP-Archiv auswählen Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Wählen Sie die Verzweigungen aus, um die Änderungen zu sehen und anzunehmen oder zu verwerfen -Set\ field=Setze Feld -Set\ fields=Felder setzen - Settings=Einstellungen Shortcut=Tastenkürzel @@ -825,7 +834,12 @@ Size=Größe Skipped\ -\ No\ PDF\ linked=Übersprungen - Kein PDF verlinkt Skipped\ -\ PDF\ does\ not\ exist=Übersprungen - PDF exisitert nicht -Skipped\ entry.=Eintrag übersprungen. +JabRef\ skipped\ the\ entry.=JabRef hat den Eintrag übersprungen. +Import\ error=Fehler beim Importieren +Open\ library\ error=Fehler beim Öffnen der Bibliothek +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Bitte überprüfen Sie Ihre Bibliotheksdatei auf falsche Syntax. +SourceTab\ error=SourceTab Fehler +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Benutzereingabe via Eintragseditor im Tab `{}bibtex source` führte zu einem Fehler. Sort\ subgroups=Untergruppen sortieren @@ -972,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Zeile %0\: Be No\ full\ text\ document\ found=Kein Volltext-Dokument gefunden Download\ from\ URL=Download von URL Rename\ field=Feld umbenennen -Append\ field=An Feld anfügen -Append\ to\ fields=An Felder anfügen -Rename\ field\ to=Feld umbenennen -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Inhalt eines Felds in ein Feld mit anderem Namen verschieben Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Port %0 konnte nicht für externen Zugriff genutzt werden; er wird möglicherweise von einer anderen Anwendung benutzt. Versuchen Sie einen anderen Port. @@ -1243,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Benutzerdefinierte Proxy-Konfiguration verwend Proxy\ requires\ authentication=Proxy erfordert Authentifizierung Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Achtung\: Das Passwort wird im Klartext gespeichert\! Clear\ connection\ settings=Verbindungseinstellungen zurücksetzen +Check\ Proxy\ Setting=Überprüfe Proxy-Einstellungen +Check\ connection=Verbindung prüfen +Connection\ failed\!=Verbindung fehlgeschlagen\! +Connection\ successful\!=Verbindung erfolgreich\! + +SSL\ Configuration=SSL-Konfiguration +SSL\ configuration\ changed=SSL-Konfiguration geändert +SSL\ certificate\ file=SSL-Zertifikatsdatei +Duplicate\ Certificates=Doppelte Zertifikate +You\ already\ added\ this\ certificate=Sie haben dieses Zertifikat bereits hinzugefügt Open\ folder=Ordner öffnen Export\ sort\ order=Sortierreihenfolge exportieren @@ -1279,6 +1299,7 @@ Please\ open\ %0\ manually.=Bitte öffnen Sie %0 manuell. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Der Link wurde in die Zwischenablage kopiert. Open\ %0\ file=%0 Datei öffnen +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Konsole konnte nicht automatisch erkannt werden. Bitte definieren Sie eine eigene Konsolenapplikation in den Einstellungen. Cannot\ delete\ file=Datei kann nicht gelöscht werden. File\ permission\ error=Fehler beim Dateizugriff. @@ -1666,6 +1687,8 @@ Issue\ report\ successful=Problemreport war erfolgreich Your\ issue\ was\ reported\ in\ your\ browser.=Ihr Problem wurde im Browser gemeldet. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Die Log und Ausnahmefehler-Informationen wurden in die Zwischenablage kopiert. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Bitte fügen Sie die Informationen (mit STRG+V) in die Problembeschreibung ein. +Last\ notification=Letzte Benachrichtigung +Check\ the\ event\ log\ to\ see\ all\ notifications=Überprüfe das Ereignisprotokoll, um alle Benachrichtigungen zu sehen Host=Host Port=Port @@ -1703,7 +1726,6 @@ Protected\ terms\ file=Geschützte Begriffsdatei Style\ file=Stildatei Open\ OpenOffice/LibreOffice\ connection=Öffne OpenOffice/LibreOffice Verbindung -You\ must\ enter\ at\ least\ one\ field\ name=Sie müssen mindestens einen Feldnamen angeben Non-ASCII\ encoded\ character\ found=Nicht ASCII-kodiertes Zeichen gefunden Non-UTF-8\ encoded\ field\ found=Nicht UTF-8 kodiertes Feld gefunden Toggle\ web\ search\ interface=Websuche ein-/ausschalten @@ -1841,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Entfernen aller Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Konnte Daten von '%0' nicht abrufen. Entry\ from\ %0\ could\ not\ be\ parsed.=Eintrag von %0 konnte nicht analysiert werden. Invalid\ identifier\:\ '%0'.=Ungültige Kennung\: "%0". -This\ paper\ has\ been\ withdrawn.=Dieses Paper wurde zurückgezogen. empty\ citation\ key=leerer Zitationsschlüssel Aux\ file=Aux-Datei Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Gruppe mit Einträgen die in einer vorhanden Tex-Datei zitiert worden sind @@ -2324,10 +2345,6 @@ Removes\ digits.=Ziffern entfernen. Presets=Voreinstellungen -Check\ Proxy\ Setting=Überprüfe Proxy-Einstellungen -Check\ connection=Verbindung prüfen -Connection\ failed\!=Verbindung fehlgeschlagen\! -Connection\ successful\!=Verbindung erfolgreich\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Gruppen aus Schlüsselwörtern im folgenden Feld generieren Generate\ groups\ for\ author\ last\ names=Gruppen für Nachnamen der Autoren generieren Regular\ expression=Regulärere Ausdruck @@ -2400,7 +2417,6 @@ Entry\ Type=Eintragstyp Entry\ types=Eintragstypen Field\ names=Feldnamen Others=Andere -Overwrite\ existing\ field\ values=Bestehende Feldwerte überschreiben Recommended=Empfohlen Authors\ and\ Title=Autoren und Titel @@ -2445,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Remote-Dienste Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Erlaube das Senden von PDF-Dateien und Rohzitaten an einen JabRef Online-Dienst (Grobid) um Metadaten zu ermitteln. Dies führt zu besseren Ergebnissen. +Fetcher\ cannot\ be\ tested\!=Der Fetcher kann nicht getestet werden\! +Fetcher\ unknown\!=Fetcher unbekannt\! Character\ by\ character=Zeichen für Zeichen Embedded=Eingebettet @@ -2468,26 +2486,32 @@ Valid\ from=Gültig seit Valid\ to=Gültig bis Signature\ algorithm=Signaturalgorithmus Version=Version -SSL\ Configuration=SSL-Konfiguration -SSL\ certificate\ file=SSL-Zertifikatsdatei -Duplicate\ Certificates=Doppelte Zertifikate -You\ already\ added\ this\ certificate=Sie haben dieses Zertifikat bereits hinzugefügt Error\ downloading=Fehler beim Herunterladen Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Fehler beim Schreiben von Metadaten. Details finden Sie im Fehlerprotokoll. Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metadaten können nicht geschrieben werden, Datei %1 nicht gefunden. Success\!\ Finished\ writing\ metadata.=Erfolgreich\! Das Schreiben der Metadaten ist abgeschlossen. -Overwrite\ Non\ empty\ fields=Overwrite Non empty fields -Set=Set -Append=Append -Clear\ field=Clear field -field\ value=field value -Edit\ field\ value\ of\ selected\ entries=Edit field value of selected entries -Rename=Rename -new\ field\ name=new field name -Copy\ value=Copy value -Move\ value=Move value -Swap\ values=Swap values -Copy\ or\ move\ the\ value\ of\ one\ field\ to\ another=Copy or move the value of one field to another +Custom\ API\ key=Eigener API-Schlüssel +Check\ %0\ API\ Key\ Setting=%0 API-Schlüsseleinstellungen überprüfen + +Edit\ content=Inhalt ändern +Copy\ or\ Move\ content=Inhalt kopieren oder verschieben +Overwrite\ field\ content=Inhalt des Feldes überschreiben +Set=Festlegen +Append=Anfügen +Clear\ field\ content=Feldinhalt löschen +Set\ or\ append\ content=Setze oder füge Inhalt an +Edit\ field\ content\ for\ selected\ entries=Feldinhalt der ausgewählten Einträge bearbeiten +Rename=Umbenennen +New\ field\ name=Neuer Feldname +Copy\ content=Inhalt kopieren +Move\ content=Inhalt verschieben +Swap\ content=Inhalte vertauschen +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Inhalt eines Feldes kopieren oder in ein anderes Feld verschieben +Automatic\ field\ editor=Automatischer Feldeditor + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Hinweis\: Wenn es den ursprünglichen Einträgen an Schlüsselwörtern fehlt, die sich für die neue Gruppenkonfiguration qualifizieren, wird die Bestätigung hier diese hinzufügen) +Assign=Zuweisen +Do\ not\ assign=Nicht zuweisen diff --git a/src/main/resources/l10n/JabRef_el.properties b/src/main/resources/l10n/JabRef_el.properties index 290a41f97b6..e3f04f56ffe 100644 --- a/src/main/resources/l10n/JabRef_el.properties +++ b/src/main/resources/l10n/JabRef_el.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 ταιριάζει την συνήθη έκφραση %1 %0\ matches\ the\ term\ %1=%0 ταιριάζει τον όρο %1 @@ -126,8 +127,6 @@ Cite\ command=Εντολή αναφοράς Clear=Καθαρισμός -Clear\ fields=Καθαρισμός πεδίων - Close\ dialog=Κλείσμο διαλόγου @@ -608,11 +607,18 @@ Remove\ selected\ entries\ from\ this\ group=Αφαίρεση επιλεγμέν Remove\ group=Αφαίρεση ομάδας -Remove\ group\ "%0"?=Αφαίρεση της ομάδας "%0"; +Remove\ group\ and\ subgroups=Αφαίρεση ομάδας και υπο-ομάδων + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Αφαίρεση της ομάδας "%0" και των υπο-ομάδων της; -Remove\ group\ and\ subgroups=Αφαίρεση ομάδας και υπο-ομάδων +Removed\ group\ "%0"\ and\ its\ subgroups.=Η ομάδα "%0" και οι υπο-ομάδες της έχουν αφαιρεθεί. + + Remove\ link=Αφαίρεση συνδέσμου @@ -621,8 +627,6 @@ Remove\ old\ entry=Αφαίρεση παλαιάς καταχώρησης Removed\ group\ "%0".=Η ομάδα "%0" έχει αφαιρεθεί. -Removed\ group\ "%0"\ and\ its\ subgroups.=Η ομάδα "%0" και οι υπο-ομάδες της έχουν αφαιρεθεί. - Removed\ string=Η συμβολοσειρά έχει αφαιρεθεί Renamed\ string=Η συμβολοσειρά έχει μετονομαστεί @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Επιλογή αρχείου από συμπι Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Επιλέξτε τους δενδροειδείς κόμβους για να προβάλλετε και να αποδεχτείτε ή να απορρίψετε αλλαγές -Set\ field=Ορισμός πεδίου -Set\ fields=Ορισμός πεδίων - Settings=Ρυθμίσεις Shortcut=Συντόμευση @@ -714,7 +715,6 @@ Size=Μέγεθος Skipped\ -\ No\ PDF\ linked=Παραβλέφθηκε - Δεν έχει συνδεθεί PDF Skipped\ -\ PDF\ does\ not\ exist=Παραβλέφθηκε - Δεν υπάρχει PDF -Skipped\ entry.=Η καταχώρηση παραβλέφθηκε. source\ edit=επεξεργασία πηγής @@ -829,10 +829,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru No\ full\ text\ document\ found=Δεν βρέθηκε αρχείο πλήρους κειμένου Download\ from\ URL=Κατέβασμα από σύνδεσμο URL Rename\ field=Μετονομασία πεδίου -Append\ field=Προσάρτηση πεδίου -Append\ to\ fields=Προσάρτηση στα πεδία -Rename\ field\ to=Μετονομασία πεδίου ως -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Μετακίνηση περιεχομένων ενός πεδίου σε ένα πεδίο με διαφορετικό όνομα Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Αδύνατη η χρήση της θύρας %0 για απομακρυσμένη λειτουργία, ίσως χρησιμοποιείται από μια άλλη εφαρμογή. Δοκιμάστε να ορίσετε άλλη θύρα. @@ -1046,6 +1042,7 @@ Proxy\ requires\ authentication=Ο διακομιστής μεσολάβησης Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Προσοχή\: Ο κωδικός πρόσβασης είναι αποθηκευμένος σε απλό κείμενο\! Clear\ connection\ settings=Εκκαθάριση ρυθμίσεων σύνδεσης + Open\ folder=Άνοιγμα φακέλου Export\ sort\ order=Εξαγωγή σειράς ταξινόμησης @@ -1448,7 +1445,6 @@ Protected\ terms\ file=Αρχείο προστατευμένων όρων Style\ file=Αρχείο στυλ Open\ OpenOffice/LibreOffice\ connection=Άνοιγμα σύνδεσης OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Πρέπει να καταχωρήσετε τουλάχιστον ένα όνομα πεδίου Non-ASCII\ encoded\ character\ found=Βρέθηκε χαρακτήρας χωρίς κωδικοποίηση ASCII Toggle\ web\ search\ interface=Εναλλαγή διεπαφής αναζήτησης στο διαδίκτυο @@ -1569,7 +1565,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Αφαιρεί Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Αδυναμία ανάκτησης δεδομένων καταχώρησης από '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Αδυναμία ανάλυσης καταχώρησης από %0. Invalid\ identifier\:\ '%0'.=Μη έγκυρο αναγνωριστικό\: '%0'. -This\ paper\ has\ been\ withdrawn.=Αυτή η εργασία έχει αποσυρθεί. Aux\ file=Αρχείο AUX Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Ομάδα που περιέχει καταχωρήσεις που αναφέρονται σε ένα συγκεκριμένο αρχείο TeX @@ -1702,6 +1697,8 @@ Default\ pattern=Προεπιλεγμένο μοτίβο + + diff --git a/src/main/resources/l10n/JabRef_en.properties b/src/main/resources/l10n/JabRef_en.properties index 5ae7dae2bab..ab98eefe526 100644 --- a/src/main/resources/l10n/JabRef_en.properties +++ b/src/main/resources/l10n/JabRef_en.properties @@ -833,7 +833,12 @@ Size=Size Skipped\ -\ No\ PDF\ linked=Skipped - No PDF linked Skipped\ -\ PDF\ does\ not\ exist=Skipped - PDF does not exist -Skipped\ entry.=Skipped entry. +JabRef\ skipped\ the\ entry.=JabRef skipped the entry. +Import\ error=Import error +Open\ library\ error=Open library error +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Please check your library file for wrong syntax. +SourceTab\ error=SourceTab error +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=User input via entry-editor in `{}bibtex source` tab led to failure. Sort\ subgroups=Sort subgroups @@ -2484,20 +2489,20 @@ Success\!\ Finished\ writing\ metadata.=Success! Finished writing metadata. Custom\ API\ key=Custom API key Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting -Edit\ field\ value=Edit field value -Two\ fields=Two fields -Overwrite\ Non\ empty\ fields=Overwrite Non empty fields +Edit\ content=Edit content +Copy\ or\ Move\ content=Copy or Move content +Overwrite\ field\ content=Overwrite field content Set=Set Append=Append -Clear\ field=Clear field -Field\ value=Field value -Edit\ field\ value\ of\ selected\ entries=Edit field value of selected entries +Clear\ field\ content=Clear field content +Set\ or\ append\ content=Set or append content +Edit\ field\ content\ for\ selected\ entries=Edit field content for selected entries Rename=Rename New\ field\ name=New field name -Copy\ value=Copy value -Move\ value=Move value -Swap\ values=Swap values -Copy\ or\ move\ the\ value\ of\ one\ field\ to\ another=Copy or move the value of one field to another +Copy\ content=Copy content +Move\ content=Move content +Swap\ content=Swap content +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copy or move the content of one field to another Automatic\ field\ editor=Automatic field editor Open\ Link=Open Link diff --git a/src/main/resources/l10n/JabRef_es.properties b/src/main/resources/l10n/JabRef_es.properties index c70107490a3..60e35d98ad0 100644 --- a/src/main/resources/l10n/JabRef_es.properties +++ b/src/main/resources/l10n/JabRef_es.properties @@ -1,7 +1,12 @@ +Could\ not\ delete\ empty\ entries.=No se pudieron eliminar entradas vacías. +Delete\ empty\ entries=Eliminar entradas vacías +Empty\ entries=Vaciar entradas +Keep\ empty\ entries=Mantener entradas vacías +Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=La biblioteca '%0' tiene entradas vacías. ¿Quieres eliminarlas? Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ restart.\ You\ may\ encounter\ errors\ if\ you\ continue\ with\ this\ session.=No es posible supervisar los cambios en los archivos. Cierre los archivos y procesos y reinicie. Puede que se produzcan errores si continúa con esta sesión. %0\ contains\ the\ regular\ expression\ %1=%0 contiene la expresión regular %1 @@ -15,6 +20,8 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entradas +Reveal\ in\ File\ Explorer=Mostrar en Explorador de Archivos + %0\ matches\ the\ regular\ expression\ %1=%0 coincidencias con la Expresión Regular %1 %0\ matches\ the\ term\ %1=%0 coincidencias con el término %1 @@ -48,6 +55,7 @@ The\ path\ need\ not\ be\ on\ the\ classpath\ of\ JabRef.=La ruta no debe estar Add\ a\ regular\ expression\ for\ the\ key\ pattern.=Añadir una expresión regular para el patrón clave. +Add\ entry\ manually=Añadir entrada manualmente Add\ selected\ entries\ to\ this\ group=Añadir entradas seleccionadas a este grupo @@ -144,8 +152,6 @@ Cite\ command=Comando Citar Clear=Limpiar -Clear\ fields=Limpiar campos - Open\ /\ close\ entry\ editor=Abrir o cerrar editor de entradas Close\ dialog=Cerrar diálogo @@ -167,6 +173,8 @@ Copy=Copiar Copy\ title=Copiar título Copy\ \\cite{citation\ key}=Copiar \\cite{citation key} +Copy\ citation\ (html)=Copiar cita (html) +Copy\ citation\ (text)=Copiar cita (texto) Copy\ citation\ key=Copiar clave de cita Copy\ citation\ key\ and\ link=Copiar clave de cita y enlace Copy\ citation\ key\ and\ title=Copiar clave y título de cita @@ -215,13 +223,17 @@ cut\ entries=Cortar entradas cut\ entry\ %0=corte de entrada %0 +DOI\ not\ found=DOI no encontrado Library\ encoding=Codificación de la biblioteca Library\ properties=Propiedades de la biblioteca +%0\ -\ Library\ properties=%0 - Propiedades de la biblioteca Default=Por defecto +Character\ encoding\ UTF-8\ is\ not\ supported.=La codificación de caracteres UTF-8 no restá soportada. +UTF-8\ could\ not\ be\ used\ to\ encode\ the\ following\ characters\:\ %0=UTF-8 no se pudo utilizar para codificar los siguientes caracteres\: %0 The\ chosen\ encoding\ '%0'\ could\ not\ encode\ the\ following\ characters\:=La codificación de caracteres '%' no puede codificar los siguientes caracteres\: Downloading=Descargando @@ -270,6 +282,7 @@ Downloaded\ website\ as\ an\ HTML\ file.=Se descargó el sitio web como un archi duplicate\ removal=eliminación de duplicados +Duplicate\ fields=Campos duplicados Duplicate\ string\ name=Nombre de cadena duplicado @@ -331,6 +344,7 @@ External\ file\ links=Enlaces a archivos externos External\ programs=Programas externos +Failed\ to\ import\ by\ ID=Error al importar por ID Field=Campo @@ -355,6 +369,7 @@ Filter=Filtro Filter\ groups=Filtros +Finished\ writing\ metadata\ for\ %0\ file\ (%1\ skipped,\ %2\ errors).=Escritura de metadatos para el archivo %0 finalizada (%1 omitidos, %2 errores). First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=En primer lugar, seleccione las entradas para las que desea generar claves @@ -371,6 +386,7 @@ Formatter\ name=Nombre del formateador found\ in\ AUX\ file=encontrado en archivo AUX +Fulltext\ search=Búsqueda de texto completo Fulltext\ for=Texto completo de @@ -449,6 +465,9 @@ Include\ subgroups\:\ When\ selected,\ view\ entries\ contained\ in\ this\ group Independent\ group\:\ When\ selected,\ view\ only\ this\ group's\ entries=Grupo independiente\: ver sólo las entradas de este grupo cuando esté seleccionado. I\ Agree=Acepto +Indexing\ pdf\ files=Indexando archivos pdf +Indexing\ for\ %0=Indexando para %0 +%0\ of\ %1\ linked\ files\ added\ to\ the\ index=%0 de %1 archivos enlazados añadidos al índice Invalid\ citation\ key=La clave de cita no es válida @@ -462,6 +481,8 @@ JabRef\ requests\ recommendations\ from\ Mr.\ DLib,\ which\ is\ an\ external\ se JabRef\ Version\ (Required\ to\ ensure\ backwards\ compatibility\ with\ Mr.\ DLib's\ Web\ Service)=Versión JabRef (necesaria para asegurar la compatibilidad con el servicio web de Mr. DLib) Journal\ abbreviations=Abreviaturas de publicaciones +Journal\ lists\:=Listados de revistas\: +Remove\ journal\ '%0'=Eliminar revista '%0' Keep\ both=Mantener ambos @@ -497,6 +518,7 @@ Main\ file\ directory=Carpeta del archivo principal Manage\ custom\ exports=Administrar exportaciones personalizadas Manage\ custom\ imports=Administrar importaciones personalizadas +External\ file\ types=Tipos de archivos externos Mark\ new\ entries\ with\ owner\ name=Marcar nuevas entradas con nombre de propietario @@ -519,7 +541,9 @@ Moved\ group\ "%0".=Se ha movido el grupo "%0". Mr.\ DLib\ Privacy\ settings=Mr. DLib Configuración de la privacidad +No\ database\ is\ open=No hay ninguna base de datos abierta +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=Necesitamos una base de datos desde la que exportar. Abrir una. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=No se han recibido recomendaciones del Mr. DLib para esta entrada. @@ -553,6 +577,7 @@ No\ journal\ names\ could\ be\ abbreviated.=No se pudieron abreviar nombres de r No\ journal\ names\ could\ be\ unabbreviated.=No se pudieron expandir nombres de revistas. +No\ DOI\ data\ exists=No existen datos DOI not=no @@ -677,11 +702,25 @@ Remove\ selected\ entries\ from\ this\ group=Eliminar las entradas seleccionadas Remove\ group=Eliminar grupo -Remove\ group\ "%0"?=¿Eliminar el grupo "%0%? +Remove\ group\ and\ subgroups=Eliminar grupo y subgrupos + +Remove\ groups\ and\ subgroups=Eliminar grupos y subgrupos + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=¿Eliminar todos los grupos seleccionados y mantener sus subgrupos? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=¿Eliminar grupo "%0" y mantener sus subgrupos? + +Remove\ groups=Eliminar grupos + +Removed\ all\ selected\ groups.=Eliminados todos los grupos seleccionados. Remove\ group\ "%0"\ and\ its\ subgroups?=¿Eliminar el grupo "%0" y sus subgrupos? -Remove\ group\ and\ subgroups=Eliminar grupo y subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Se ha eliminado el grupo "%0" y sus subgrupos. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=¿Eliminar todos los grupos seleccionados y sus subgrupos? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Eliminados todos los grupos seleccionados y sus subgrupos. Remove\ link=Eliminar enlace @@ -691,8 +730,6 @@ Remove\ string\ %0=Eliminar cadena %0 Removed\ group\ "%0".=el grupo "%0" se ha eliminado. -Removed\ group\ "%0"\ and\ its\ subgroups.=Se ha eliminado el grupo "%0" y sus subgrupos. - Removed\ string=Cadena eliminada Renamed\ string=Cadena renombrada @@ -715,6 +752,8 @@ Replaces\ Unicode\ ligatures\ with\ their\ expanded\ form=Reemplaza las ligadura Required\ fields=Campos requeridos +Do\ not\ resolve\ BibTeX\ strings=No resolver cadenas BibTeX +Resolve\ BibTeX\ strings\ for\ the\ following\ fields=Resolver las cadenas BibTeX para los siguientes campos resolved=resuelto @@ -722,11 +761,13 @@ Restart=Reiniciar Restart\ required=Reinicio requerido +Return\ to\ dialog=Volver al diálogo Review=Revisar Review\ changes=Revisar cambios Review\ Field\ Migration=Revisar campo de migración +Loading=Cargando Save=Guardar Save\ all\ finished.=Guardar todos los finalizados @@ -740,6 +781,7 @@ Save\ library\ as...=Guardar biblioteca como... Saving=Guardando Saving\ all\ libraries...=Guardando todas las bibliotecas... Saving\ library=Guardando biblioteca +Library\ saved=Biblioteca guardada Saved\ selected\ to\ '%0'.=Selección guardada en '%0'. Search=Buscar @@ -759,9 +801,6 @@ Select\ file\ from\ ZIP-archive=Seleccionar archivo desde archivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Seleccionar nodos de árbol para ver y aceptar o rechazar los cambios. -Set\ field=Establecer campo -Set\ fields=Establecer campos - Settings=Ajustes Shortcut=Atajo @@ -794,7 +833,12 @@ Size=Tamaño Skipped\ -\ No\ PDF\ linked=Omitido - No se enlazó PDF Skipped\ -\ PDF\ does\ not\ exist=Omitido - No existe el PDF -Skipped\ entry.=Entrada omitida. +JabRef\ skipped\ the\ entry.=JabRef omitió la entrada. +Import\ error=Error al importar +Open\ library\ error=Error al abrir la biblioteca +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Por favor, comprueba errores de sintaxis en tu archivo de biblioteca. +SourceTab\ error=Error de SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Entrada de usuario a través del editor de entrada en la pestaña `{}bibtex source` condujo al error. Sort\ subgroups=Ordenar subgrupos @@ -883,13 +927,18 @@ Warning=Advertencia Warnings=Advertencias +Warning\:\ You\ added\ field\ "%0"\ twice.\ Only\ one\ will\ be\ kept.=Advertencia\: Añadiste el campo "%0" dos veces. Solo uno será utilizado. web\ link=enlace a web What\ do\ you\ want\ to\ do?=¿Qué desea hacer? Whatever\ option\ you\ choose,\ Mr.\ DLib\ may\ share\ its\ data\ with\ research\ partners\ to\ further\ improve\ recommendation\ quality\ as\ part\ of\ a\ 'living\ lab'.\ Mr.\ DLib\ may\ also\ release\ public\ datasets\ that\ may\ contain\ anonymized\ information\ about\ you\ and\ the\ recommendations\ (sensitive\ information\ such\ as\ metadata\ of\ your\ articles\ will\ be\ anonymised\ through\ e.g.\ hashing).\ Research\ partners\ are\ obliged\ to\ adhere\ to\ the\ same\ strict\ data\ protection\ policy\ as\ Mr.\ DLib.=Sea cual sea la opción que elija, Mr. DLib puede compartir sus datos con socios de investigación para mejorar aún más la calidad de las recomendaciones como parte de un "laboratorio vivo". Mr. DLib también puede publicar conjuntos de datos públicos que pueden contener información anónima sobre usted y las recomendaciones (la información confidencial como los metadatos de sus artículos será anonimizada a través de, por ejemplo, hashing). Los socios de investigación están obligados a adherirse a la misma estricta política de protección de datos que Mr. DLib. +Will\ write\ metadata\ to\ the\ PDFs\ linked\ from\ selected\ entries.=Escribirá metadatos en los PDFs enlazados desde entradas seleccionadas. +Write\ BibTeXEntry\ as\ metadata\ to\ PDF.=Escribir entrada BibTeX como metadatos XMP en el PDF. +Write\ metadata\ for\ all\ PDFs\ in\ current\ library?=¿Escribir metadatos para todos los PDFs de la biblioteca actual? +Writing\ metadata\ for\ selected\ entries...=Escribiendo metadatos para las entradas seleccionadas... Write\ BibTeXEntry\ as\ XMP\ metadata\ to\ PDF.=Escribe BibTeXEntry como metadatos XMP en los PDF. @@ -929,10 +978,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Renglón %0\: No\ full\ text\ document\ found=No se encontró un documento de texto completo Download\ from\ URL=Descargar desde URL Rename\ field=Renombrar campo -Append\ field=Añadir campo -Append\ to\ fields=Añadir a campos -Rename\ field\ to=Renombrar campo a -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover contenidos de un campo en un campo con nombre diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=No se puede usar el puerto %0 para operación remota\: Puede estar en uso por otra aplicación. Pruebe a especificar otro puerto. @@ -1182,6 +1227,11 @@ Use\ custom\ proxy\ configuration=Usar configuración de proxy personalizada Proxy\ requires\ authentication=El proxi requiere autenticación Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atención\: ¡La contraseña está almacenada como texto plano\! Clear\ connection\ settings=Limpiar ajustes de conexión +Check\ Proxy\ Setting=Comprobar configuración de «proxy» +Check\ connection=Comprobar conexión +Connection\ failed\!=Falló la conexión. +Connection\ successful\!=Conexión correcta. + Open\ folder=Abrir pasta Export\ sort\ order=Criterio de ordenación para exportación @@ -1586,6 +1636,8 @@ Issue\ report\ successful=Comunicación de problema satisfactoria Your\ issue\ was\ reported\ in\ your\ browser.=Su problema fue comunicado a través del navegador The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=La información de excepción y registro fue copiada a su portapapeles Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Por favor, pegue esta información (con Ctrl+V) en la descripción del problema. +Last\ notification=Última notificación +Check\ the\ event\ log\ to\ see\ all\ notifications=Revisa el registro de eventos para ver todas las notificaciones Host=Host/Servidor Port=Puerto @@ -1623,7 +1675,6 @@ Protected\ terms\ file=Archivo de términos protegidos Style\ file=Archivo de estilo Open\ OpenOffice/LibreOffice\ connection=Conexión OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Debe introducir, al menos, un nombre de campo Non-ASCII\ encoded\ character\ found=Se han encontrado caracteres con codificación no-ASCII Non-UTF-8\ encoded\ field\ found=Se encontró un campo no codificado en UTF-8 Toggle\ web\ search\ interface=Cambiar interfaz de búsqueda web @@ -1732,6 +1783,7 @@ Delete\ '%0'=Eliminar '%0' Delete\ from\ disk=Eliminar del disco duro Remove\ from\ entry=Eliminar de la entrada There\ exists\ already\ a\ group\ with\ the\ same\ name.=Ya existe un grupo con el mismo nombre. +If\ you\ use\ it,\ it\ will\ inherit\ all\ entries\ from\ this\ other\ group.=Si lo utiliza, heredará todas las entradas de este otro grupo. Copy\ linked\ file=Copiar archivo enlazado Copy\ linked\ file\ to\ folder...=Copiar archivo enlazado a la carpeta... @@ -1760,7 +1812,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Elimina todos l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=No se pueden recuperar los datos de la entrada desde '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=No se pudo analizar la entrada de %0. Invalid\ identifier\:\ '%0'.=Identificador no válido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Este artículo ha sido retirado. empty\ citation\ key=clave de cita vacía Aux\ file=Archivo Aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo que contiene las entradas citadas en un archivo TeX determinado @@ -1788,6 +1839,7 @@ Could\ not\ connect\ to\ Vim\ server.\ Make\ sure\ that\ Vim\ is\ running\ with\ Could\ not\ connect\ to\ a\ running\ gnuserv\ process.\ Make\ sure\ that\ Emacs\ or\ XEmacs\ is\ running,\ and\ that\ the\ server\ has\ been\ started\ (by\ running\ the\ command\ 'server-start'/'gnuserv-start').=No se puede conectar con un proceso gnuserv en ejecución. Asegúrese de que Emacs o XEmacs se está ejecutando y de que el servidor ha sido iniciado (ejecutando el comando 'server-start'/'gnuserv-start'). Error\ pushing\ entries=Error al enviar entradas +Preamble=Preámbulo Markings=Marcados Use\ selected\ instance=Usar la instancia seleccionada @@ -1806,6 +1858,7 @@ Blog=Blog Check\ integrity=Verificar integridad Cleanup\ URL\ link=Limpiar un enlace URL Cleanup\ URL\ link\ by\ removing\ special\ symbols\ and\ extracting\ simple\ link=Limpiar un enlace URL eliminando los símbolos especiales y extrayendo un enlace simple +Copy\ DOI=Copiar DOI Copy\ DOI\ url=Copiar la url del DOI Development\ version=Versión de desarrollo Export\ selected\ entries=Exportar registros seleccionados @@ -1815,6 +1868,8 @@ JabRef\ resources=Recursos sobre JabRef Manage\ journal\ abbreviations=Administrar abreviaturas de publicaciones Manage\ protected\ terms=gestionar términos protegidos New\ entry\ from\ plain\ text=Nueva entrada desde texto sin formato +Import\ by\ ID=Importar por ID +Enter\ a\ valid\ ID=Introduzca un ID válido New\ sublibrary\ based\ on\ AUX\ file=Nueva subbiblioteca a partir de un archivo AUX Push\ entries\ to\ external\ application\ (%0)=Agregar registros a aplicación externa (%0) Quit=Salir @@ -1874,6 +1929,7 @@ Keyword\ separator=Separador de palabras clave Remove\ keyword=Eliminar palabra clave Are\ you\ sure\ you\ want\ to\ remove\ keyword\:\ "%0"?=¿Está seguro de que desea eliminar la palabra clave\: "%0"? Reset\ to\ default=Restablecer los valores por defecto +String\ constants=Constantes de cadena Export\ all\ entries=Exportar todas las entradas Generate\ citation\ keys=Generar claves de cita Manage\ field\ names\ &\ content=Gestionar nombres y contenido de los campos @@ -2003,6 +2059,7 @@ Please\ provide\ a\ valid\ aux\ file.=Por favor, proporcione un archivo AUX vál Keyword\ delimiter=Separador de palabras clave Hierarchical\ keyword\ delimiter=Separador de palabras clave jerárquicas Escape\ ampersands=Escape de ampersands +Escape\ dollar\ sign=Símbolo de escape dólar Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Consejo\:\n\nPara buscar Pedro en todos los campos, escriba\:\npedro\n\nPara buscar Pedro en el campo author y eléctrico en el campo title, escriba\:\nauthor\=Pedro and title\=eléctrico @@ -2212,7 +2269,11 @@ Reveal\ in\ file\ explorer=Revelar en el explorador de archivos Autolink\ files=Enlazar archivos automáticamente +Custom\ editor\ tabs=Pestañas de editor personalizadas +Custom\ export\ formats=Formatos de exportación personalizados +Custom\ import\ formats=Formatos de importación personalizados +No\ list\ enabled=No hay lista habilitada Protect\ selection=Proteger selección Customized\ preview\ style=Estilo de previsualización personalizado @@ -2233,17 +2294,23 @@ Removes\ digits.=Elimina dígitos. Presets=Preconfiguraciones -Check\ Proxy\ Setting=Comprobar configuración de «proxy» -Check\ connection=Comprobar conexión -Connection\ failed\!=Falló la conexión. -Connection\ successful\!=Conexión correcta. Generate\ groups\ from\ keywords\ in\ the\ following\ field=Generar grupos a partir de palabras clave en el campo siguiente Generate\ groups\ for\ author\ last\ names=Generar grupos para apellidos de autores Regular\ expression=Expresión regular Error\ importing.\ See\ the\ error\ log\ for\ details.=Error al importar. Consulte el registro de errores para obtener detalles. +Error\ from\ import\:\ %0=Error de importación\: %0 +Error\ reading\ PDF\ content\:\ %0=Error al leer el contenido PDF\: %0 +Importing\ bib\ entry=Importando entrada bibliográfica +Importing\ using\ extracted\ PDF\ data=Importación usando datos extraídos del PDF +No\ BibTeX\ data\ found.\ Creating\ empty\ entry\ with\ file\ link=No se encontraron datos BibTeX. Creando entrada vacía con enlace de archivo +No\ metadata\ found.\ Creating\ empty\ entry\ with\ file\ link=No se encontraron metadatos. Creando entrada vacía con enlace de archivo +Processing\ file\ %0=Procesando archivo %0 +Export\ selected=Exportar seleccionados +Separate\ merged\ citations=Citas fusionadas separadas +Separate\ citations=Citas separadas Custom\ DOI\ URI=URI personalizado de DOI @@ -2272,7 +2339,6 @@ Entry\ Type=Tipo de entrada Entry\ types=Tipos de entrada Field\ names=Nombre del campo Others=Otros -Overwrite\ existing\ field\ values=Sobreescribir valores de campo existentes Recommended=Recomendado Authors\ and\ Title=Autoría y título @@ -2303,3 +2369,5 @@ Question=Pregunta + + diff --git a/src/main/resources/l10n/JabRef_fa.properties b/src/main/resources/l10n/JabRef_fa.properties index 9fabe5b9655..fb60c5076e1 100644 --- a/src/main/resources/l10n/JabRef_fa.properties +++ b/src/main/resources/l10n/JabRef_fa.properties @@ -14,6 +14,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ + Abbreviate\ names=نام های مخفف Abbreviated\ %0\ journal\ names.=نام های مخفف شده %0 ژورنال. @@ -98,8 +99,6 @@ Case\ sensitive=حساس به کوچکی و بزرگی حروف Clear=پاکسازی -Clear\ fields=پاک کردن فیلدها - @@ -375,13 +374,18 @@ Open\ terminal\ here=در اینجا پایانه را باز کن -Library\ saved=کتابخانه ذخیره شد +Library\ saved=کتابخانه ذخیره شد + + + + + @@ -496,6 +500,7 @@ Merge\ entries=ترکیب کردن ورودیها + Open\ folder=بازکردن پوشه @@ -665,3 +670,5 @@ Search\ results\ from\ open\ libraries=جستجوی نتایج از کتابخا + + diff --git a/src/main/resources/l10n/JabRef_fr.properties b/src/main/resources/l10n/JabRef_fr.properties index 425843db382..46481f71dce 100644 --- a/src/main/resources/l10n/JabRef_fr.properties +++ b/src/main/resources/l10n/JabRef_fr.properties @@ -19,7 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entrées -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=Exportation de %0 réussie. Voulez-vous ouvrir le dossier contenant le fichier sauvegardé ? +Export\ operation\ finished\ successfully.=L'opération d'export s'est terminée avec succès. + +Reveal\ in\ File\ Explorer=Montrer dans l'explorateur de fichiers %0\ matches\ the\ regular\ expression\ %1=%0 correspond à l'expression régulière %1 @@ -151,8 +153,6 @@ Cite\ command=Commande Cite Clear=Vider -Clear\ fields=Vider les champs - Open\ /\ close\ entry\ editor=Ouvrir/fermer l'éditeur d'entrées Close\ dialog=Fermer la fenêtre @@ -295,7 +295,7 @@ Dynamically\ group\ entries\ by\ searching\ a\ field\ for\ a\ keyword=Grouper dy Each\ line\ must\ be\ of\ the\ following\ form\:\ 'tab\:field1;field2;...;fieldN'.=Chaque ligne doit être au format suivant \: 'tab\:field1;field2;...;fieldN'. -Edit=Éditer +Edit=Édition Edit\ file\ type=Modifier le type de fichier @@ -703,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Supprimer de ce groupe les entrées Remove\ group=Supprimer le groupe -Remove\ group\ "%0"?=Supprimer le groupe « %0 » ? +Remove\ group\ and\ subgroups=Supprimer le groupe et les sous-groupes + +Remove\ groups\ and\ subgroups=Supprimer les groupes et les sous-groupes + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Supprimer tous les groupes sélectionnés et conserver leurs sous-groupes ? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Supprimer le groupe « %0 » et conserver ses sous-groupes ? + +Remove\ groups=Supprimer les groupes + +Removed\ all\ selected\ groups.=Tous les groupes sélectionnés ont été supprimés. Remove\ group\ "%0"\ and\ its\ subgroups?=Supprimer le groupe « %0 » et ses sous-groupes ? -Remove\ group\ and\ subgroups=Supprimer le groupe et les sous-groupes +Removed\ group\ "%0"\ and\ its\ subgroups.=Groupe « %0 » et ses sous-groupes supprimés. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Supprimer tous les groupes sélectionnés et leurs sous-groupes ? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Tous les groupes sélectionnés et leurs sous-groupes ont été supprimés. Remove\ link=Supprimer le lien @@ -717,8 +731,6 @@ Remove\ string\ %0=Supprimer la chaîne %0 Removed\ group\ "%0".=Groupe « %0 » supprimé. -Removed\ group\ "%0"\ and\ its\ subgroups.=Groupe « %0 » et ses sous-groupes supprimés. - Removed\ string=Chaîne supprimée Renamed\ string=Chaîne renommée @@ -770,7 +782,7 @@ Save\ library\ as...=Enregistrer le fichier sous... Saving=Enregistrement en cours Saving\ all\ libraries...=Enregistrement de tous les fichiers... Saving\ library=Enregistrement du fichier en cours -Library\ saved=Fichier sauvé +Library\ saved=Fichier enregistré Saved\ selected\ to\ '%0'.=Sélection enregistrée dans « %0 ». Search=Recherche @@ -790,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Sélectionner un fichier depuis une archive ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Sélectionner les nœuds de l'arborescence pour voir, et accepter ou rejeter, les modifications -Set\ field=Configurer le champ -Set\ fields=Configurer les champs - Settings=Paramètres Shortcut=Raccourci @@ -825,7 +834,12 @@ Size=Taille Skipped\ -\ No\ PDF\ linked=Sauté - Pas de PDF lié Skipped\ -\ PDF\ does\ not\ exist=Omis - Le PDF n'existe pas -Skipped\ entry.=Entrée omise. +JabRef\ skipped\ the\ entry.=JabRef a ignoré l'entrée. +Import\ error=Erreur d'importation +Open\ library\ error=Erreur d'ouverture du fichier +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Veuillez vérifier la syntaxe de votre fichier bibliographique. +SourceTab\ error=Erreur de SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=La saisie de l'utilisateur via l'éditeur d'entrée dans l'onglet `{}bibtex source` a conduit à un échec. Sort\ subgroups=Trier les sous-groupes @@ -971,18 +985,14 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Ligne Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Ligne %0 \: clef de citation corrompue %1 (virgule manquante). No\ full\ text\ document\ found=Aucun texte intégral trouvé Download\ from\ URL=Télécharger depuis l'URL -Rename\ field=Renommer le champ -Append\ field=Ajouter au champ -Append\ to\ fields=Ajouter aux champs -Rename\ field\ to=Renommer le champ en -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Déplacer le contenu d'un champ vers un champ d'un nom différent +Rename\ field=Renommer Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Le port %0 ne peut pas être utilisé pour une opération à distance ; un autre logiciel pourrait être en train de l'utiliser. Essayer de spécifier un autre port. Looking\ for\ full\ text\ document...=Recherche du texte intégral... A\ local\ copy\ will\ be\ opened.=Une copie locale sera ouverte. -Autosave\ local\ libraries=Sauvegarder automatiquement les fichiers locaux -Automatically\ save\ the\ library\ to=Sauvegarder automatiquement les fichiers dans \: +Autosave\ local\ libraries=Enregistrer automatiquement les fichiers locaux +Automatically\ save\ the\ library\ to=Enregistrer automatiquement les fichiers dans \: Please\ enter\ a\ valid\ file\ path.=Veuillez saisir un chemin de fichier valide. Error\ opening\ file\ '%0'.=Erreur lors de l'ouverture du fichier « %0 ». @@ -1003,7 +1013,7 @@ contains=contient search\ expression=expression de recherche Optional\ fields\ 2=Champs optionnels 2 -Waiting\ for\ save\ operation\ to\ finish=Attente de la fin de l'opération de sauvegarde +Waiting\ for\ save\ operation\ to\ finish=Attente de la fin de l'opération d'enregistrement Waiting\ for\ background\ tasks\ to\ finish.\ Quit\ anyway?=En attente de la fin des tâches en arrière-plan. Quitter quand même ? Find\ and\ remove\ duplicate\ citation\ keys=Rechercher et supprimer les clefs de citation dupliquées @@ -1037,7 +1047,7 @@ Output\ file\ missing=Fichier de sortie manquant No\ search\ matches.=Recherche sans correspondance. The\ output\ option\ depends\ on\ a\ valid\ input\ option.=L'option de sortie dépend d'une option d'entrée valide. Linked\ file\ name\ conventions=Conventions pour les noms de fichiers liés -Filename\ format\ pattern=Modèle de format de nom de fichier +Filename\ format\ pattern=Modèle de format de nom de fichier Additional\ parameters=Paramètres additionnels Cite\ selected\ entries\ between\ parenthesis=Citer les entrées sélectionnées entre parenthèses Cite\ selected\ entries\ with\ in-text\ citation=Citer les entrées sélectionnées comme incluse dans le texte @@ -1243,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Utiliser une configuration de proxy personnali Proxy\ requires\ authentication=Le proxy nécessite une authentification Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Attention \: Le mot de passe est stocké en clair dans du texte brut \! Clear\ connection\ settings=Réinitialiser les paramètres de connexion +Check\ Proxy\ Setting=Vérifier la configuration du proxy +Check\ connection=Vérifier la connexion +Connection\ failed\!=Échec de la connexion \! +Connection\ successful\!=Connexion réussie \! + +SSL\ Configuration=Configuration SSL +SSL\ configuration\ changed=Configuration SSL modifiée +SSL\ certificate\ file=Fichier de certificat SSL +Duplicate\ Certificates=Dupliquer les certificats +You\ already\ added\ this\ certificate=Vous avez déjà ajouté ce certificat Open\ folder=Ouvrir le répertoire Export\ sort\ order=Exporter l'ordre de tri @@ -1279,6 +1299,7 @@ Please\ open\ %0\ manually.=Veuillez ouvrir manuellement %0 . The\ link\ has\ been\ copied\ to\ the\ clipboard.=Le lien a été copié dans le presse-papiers. Open\ %0\ file=Ouvrir le fichier %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Impossible de détecter le terminal automatiquement. Veuillez définir un terminal personnalisé dans les préférences. Cannot\ delete\ file=Le fichier ne peut pas être supprimé File\ permission\ error=Erreur due aux permissions du fichier @@ -1666,6 +1687,8 @@ Issue\ report\ successful=Signalement de l'anomalie réussi Your\ issue\ was\ reported\ in\ your\ browser.=Votre anomalie a été affichée dans votre navigateur. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Le journal et les informations d'anomalie ont été copiées dans votre presse-papier. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Veuillez coller ces informations (avec Ctrl+V) dans la description de l'anomalie. +Last\ notification=Dernière notification +Check\ the\ event\ log\ to\ see\ all\ notifications=Consultez le journal des événements pour voir toutes les notifications Host=Hôte Port=Port @@ -1703,7 +1726,6 @@ Protected\ terms\ file=Fichier des termes protégés Style\ file=Fichier de style Open\ OpenOffice/LibreOffice\ connection=Ouvrir une connexion OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Vous devez saisir au moins un nom de champ Non-ASCII\ encoded\ character\ found=Caractère ayant un encodage non-ASCII trouvé Non-UTF-8\ encoded\ field\ found=Un champ non encodé en UTF-8 a été trouvé Toggle\ web\ search\ interface=Afficher/Masquer l'interface de recherche internet @@ -1841,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Supprime du con Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Impossible de récupérer les données sur l'entrée depuis « %0 ». Entry\ from\ %0\ could\ not\ be\ parsed.=Entrée de %0 n’a pas pu être analysée. Invalid\ identifier\:\ '%0'.=Identifiant non valide \: « %0 ». -This\ paper\ has\ been\ withdrawn.=Cet article a été retiré. empty\ citation\ key=clef de citation vide Aux\ file=Fichier AUX Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Groupe contenant les entrées citées dans un fichier TeX spécifique @@ -2144,7 +2165,7 @@ Search\ ShortScience=Recherche ShortScience Unable\ to\ open\ ShortScience.=Impossible de se connecter à ShortScience. Shared\ database=Base de données partagée -Lookup=Rechercher +Lookup=Recherche Please\ enter\ a\ field\ name\ to\ search\ for\ a\ keyword.=Veuillez entrer le nom du champ dans lequel chercher un mot-clef. Access\ date\ of\ the\ address\ specified\ in\ the\ url\ field.=Date d'accès de l'adresse spécifiée dans le champ "url". @@ -2324,10 +2345,6 @@ Removes\ digits.=Supprime les chiffres. Presets=Préréglages -Check\ Proxy\ Setting=Vérifier la configuration du proxy -Check\ connection=Vérifier la connexion -Connection\ failed\!=Échec de la connexion \! -Connection\ successful\!=Connexion réussie \! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Créer des groupes à partir des mot-clefs du champ suivant Generate\ groups\ for\ author\ last\ names=Créer des groupes à partir des noms propres des auteurs Regular\ expression=Expression régulière @@ -2400,7 +2417,6 @@ Entry\ Type=Type d’entrée Entry\ types=Types d'entrée Field\ names=Noms de champs Others=Autres -Overwrite\ existing\ field\ values=Écraser les valeurs des champs existants Recommended=Usuels Authors\ and\ Title=Auteurs et titre @@ -2470,10 +2486,6 @@ Valid\ from=Valable à partir de Valid\ to=Valide jusqu'au Signature\ algorithm=Algorithme de signature Version=Version -SSL\ Configuration=Configuration SSL -SSL\ certificate\ file=Fichier de certificat SSL -Duplicate\ Certificates=Dupliquer les certificats -You\ already\ added\ this\ certificate=Vous avez déjà ajouté ce certificat Error\ downloading=Erreur de téléchargement @@ -2483,3 +2495,23 @@ Success\!\ Finished\ writing\ metadata.=Succès \! Écriture des métadonnées t Custom\ API\ key=Clef d'API personnalisée Check\ %0\ API\ Key\ Setting=Vérifier les paramètres de la clef d'API %0 + +Edit\ content=Modifier le contenu +Copy\ or\ Move\ content=Copier ou déplacer le contenu +Overwrite\ field\ content=Écraser le contenu du champ +Set=Définir +Append=Ajouter +Clear\ field\ content=Effacer le contenu du champ +Set\ or\ append\ content=Définir ou ajouter du contenu +Edit\ field\ content\ for\ selected\ entries=Modifier le contenu d'un champ +Rename=Renommer +New\ field\ name=Nouveau nom de champ +Copy\ content=Copier le contenu +Move\ content=Déplacer le contenu +Swap\ content=Permuter le contenu +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copier ou déplacer le contenu d'un champ vers un autre +Automatic\ field\ editor=Éditeur automatique de champs + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note \: si les entrées originales n'ont pas de mots-clefs correspondant à la nouvelle configuration du groupe, confirmer ici les ajoutera) +Assign=Assigner +Do\ not\ assign=Ne pas assigner diff --git a/src/main/resources/l10n/JabRef_id.properties b/src/main/resources/l10n/JabRef_id.properties index 0e582b7cb21..6458e06d8e0 100644 --- a/src/main/resources/l10n/JabRef_id.properties +++ b/src/main/resources/l10n/JabRef_id.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 sesuai dengan Ekspresi Reguler %1 %0\ matches\ the\ term\ %1=%0 sesuai dengan istilah %1 @@ -125,8 +126,6 @@ Cite\ command=Perintah acuan Clear=Bersihkan -Clear\ fields=Bersihkan beberapa bidang - Close\ dialog=Tutup dialog @@ -604,11 +603,18 @@ Remove\ selected\ entries\ from\ this\ group=Menghapus entri yang dipilih dari g Remove\ group=Hapus grup -Remove\ group\ "%0"?=Hapus grup "%0"? +Remove\ group\ and\ subgroups=Hapus grup dan sub-grup + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Hapus grup "%0" dan sub-grup nya? -Remove\ group\ and\ subgroups=Hapus grup dan sub-grup +Removed\ group\ "%0"\ and\ its\ subgroups.=Hapus grup "%0" dan sub-grup nya. + + Remove\ link=Hapus tautan @@ -617,8 +623,6 @@ Remove\ old\ entry=Hapus entri lama Removed\ group\ "%0".=Hapus grup "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Hapus grup "%0" dan sub-grup nya. - Removed\ string=Hapus string Renamed\ string=Ganti nama string @@ -672,9 +676,6 @@ Select\ file\ from\ ZIP-archive=Pilih berkas dari arsip ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Pilih tiga nodal untuk melihat, menerima atau menolak perubahan -Set\ field=Pilih bidang -Set\ fields=Pilih beberapa bidang - Settings=Pengaturan Shortcut=Pintasan @@ -705,7 +706,6 @@ Size=Ukuran Skipped\ -\ No\ PDF\ linked=Dilompati - Tanpa tautan PDF Skipped\ -\ PDF\ does\ not\ exist=Dilompati - PDF tidak ada -Skipped\ entry.=Entri dilompati. source\ edit=sunting sumber @@ -819,10 +819,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Unduh dari URL Rename\ field=Ganti nama bidang -Append\ field=Tambahkan bidang -Append\ to\ fields=Tambahkan ke bidang -Rename\ field\ to=Ganti nama bidang menjadi -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Pindah isi dari bidang ke bidang lain dengan nama lain Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Tidak bisa memakai port %0 untuk operasi jauh; aplikasi lain mungkin sedang menggunakan. Coba port lain. @@ -1036,6 +1032,7 @@ Proxy\ requires\ authentication=Proxy memerlukan otentikasi Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Perhatian\: Kata sandi disimpan sebagai teks biasa\! Clear\ connection\ settings=Bersihkan pengaturan koneksi + Open\ folder=Buka direktori Export\ sort\ order=Ekspor urutan penyortiran @@ -1430,7 +1427,6 @@ Protected\ terms\ file=Istilah yang dilindungi Style\ file=File gaya Open\ OpenOffice/LibreOffice\ connection=Buka koneksi OpenOffice / LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Anda harus memasukkan setidaknya satu nama field Non-ASCII\ encoded\ character\ found=Karakter yang dikodekan non-ASCII ditemukan Toggle\ web\ search\ interface=Alihkan antarmuka pencarian web @@ -1551,7 +1547,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Menghapus semua Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Tidak dapat mengambil data entri dari ' % 0 '. Entry\ from\ %0\ could\ not\ be\ parsed.=Entri dari % 0 tidak dapat diuraikan. Invalid\ identifier\:\ '%0'.=Pengenal tidak valid\: ' % 0 '. -This\ paper\ has\ been\ withdrawn.=Makalah ini telah ditarik. @@ -1649,6 +1644,8 @@ Default\ pattern=Pola bawaan + + diff --git a/src/main/resources/l10n/JabRef_it.properties b/src/main/resources/l10n/JabRef_it.properties index 6f249bb1436..fc937b51fa7 100644 --- a/src/main/resources/l10n/JabRef_it.properties +++ b/src/main/resources/l10n/JabRef_it.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 voci +Export\ operation\ finished\ successfully.=L'operazione di esportazione si è conclusa con successo. + +Reveal\ in\ File\ Explorer=Mostra nel File Explorer %0\ matches\ the\ regular\ expression\ %1=%0 corrisponde all'espressione regolare %1 @@ -150,8 +153,6 @@ Cite\ command=Comando Cite Clear=Svuota -Clear\ fields=Annulla i campi - Open\ /\ close\ entry\ editor=Apri / chiudi editor voci Close\ dialog=Chiudi la finestra di dialogo @@ -700,11 +701,25 @@ Remove\ selected\ entries\ from\ this\ group=Rimuovi da questo gruppo le voci se Remove\ group=Rimuovi gruppo -Remove\ group\ "%0"?=Rimuovere il gruppo "%0"? +Remove\ group\ and\ subgroups=Rimuovi gruppo e sottogruppi + +Remove\ groups\ and\ subgroups=Rimuovi gruppi e sottogruppi + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Rimuovere tutti i gruppi selezionati e mantenere i loro sottogruppi? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Rimuovere il gruppo "%0" e mantenere i suoi sottogruppi? + +Remove\ groups=Rimuovere i gruppi + +Removed\ all\ selected\ groups.=Rimossi tutti i gruppi selezionati. Remove\ group\ "%0"\ and\ its\ subgroups?=Rimuovere il gruppo "%0" ed i suoi sottogruppi? -Remove\ group\ and\ subgroups=Rimuovi gruppo e sottogruppi +Removed\ group\ "%0"\ and\ its\ subgroups.=Rimosso gruppo "%0" e suoi sottogruppi. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Rimuovere tutti i gruppi selezionati e i loro sottogruppi? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Rimossi tutti i gruppi selezionati e i loro sottogruppi. Remove\ link=Rimuovere il collegamento @@ -714,8 +729,6 @@ Remove\ string\ %0=Rimuovi la stringa %0 Removed\ group\ "%0".=Rimosso gruppo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Rimosso gruppo "%0" e suoi sottogruppi. - Removed\ string=Stringa rimossa Renamed\ string=Stringa rinominata @@ -787,9 +800,6 @@ Select\ file\ from\ ZIP-archive=Seleziona un file da un archivio ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selezionare i nodi dell'albero per vedere ed accettare o rifiutare le modifiche -Set\ field=Imposta il campo -Set\ fields=Imposta i campi - Settings=Parametri Shortcut=Scorciatoia @@ -822,7 +832,12 @@ Size=Dimensione Skipped\ -\ No\ PDF\ linked=Saltato - Nessun file PDF collegato Skipped\ -\ PDF\ does\ not\ exist=Saltato - Il file PDF non esiste -Skipped\ entry.=Voce saltata +JabRef\ skipped\ the\ entry.=JabRef ha saltato la voce. +Import\ error=Errore di importazione +Open\ library\ error=Errore di apertura libreria +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Controlla il file della tua libreria per sintassi sbagliata. +SourceTab\ error=Errore SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=L'input dell'utente tramite entry-editor nella scheda `{}bibtex source` ha portato al fallimento. Sort\ subgroups=Ordina sottogruppi @@ -969,10 +984,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Riga %0\: Tro No\ full\ text\ document\ found=Nessun documento di testo completo trovato Download\ from\ URL=Scarica dall'URL Rename\ field=Rinomina il campo -Append\ field=Aggiungi campo -Append\ to\ fields=Aggiungi ai campi -Rename\ field\ to=Rinomina il campo in -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Sposta il contenuto di un campo in un campo con nome diverso Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Impossibile utilizzare la porta %0 per operazioni remote; la porta potrebbe essere in uso da parte di un'altra applicazione. Provare a specificare una porta diversa. @@ -1228,6 +1239,12 @@ Use\ custom\ proxy\ configuration=Utilizza una configurazione del proxy personal Proxy\ requires\ authentication=Il proxy richiede un'autenticazione Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Attenzione\: Password salvata come testo semplice\! Clear\ connection\ settings=Reinizializza i parametri di connessione +Check\ Proxy\ Setting=Controlla Impostazioni Proxy +Check\ connection=Controlla la connessione +Connection\ failed\!=Connessione fallita\! +Connection\ successful\!=Connessione riuscita\! + +SSL\ configuration\ changed=Configurazione SSL modificata Open\ folder=Apri cartella Export\ sort\ order=Esporta il modo di ordinamento @@ -1264,6 +1281,7 @@ Please\ open\ %0\ manually.=Per favore aprire %0 manualmente. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Il link è stato copiato negli appunti. Open\ %0\ file=Apri il file %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Impossibile rilevare il terminale automaticamente. Si prega di definire un terminale personalizzato nelle preferenze. Cannot\ delete\ file=Non posso cancellare il file File\ permission\ error=Errore nei permessi del file @@ -1640,6 +1658,8 @@ Issue\ report\ successful=Segnalazione del problema avvenuta con successo Your\ issue\ was\ reported\ in\ your\ browser.=Il problema è stato segnalato nel browser. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Il log e le informazioni dell'eccezione sono stati copiati negli appunti. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Per favore, incolla questa informazione (con Ctrl+V) nella descrizione del problema. +Last\ notification=Ultima notifica +Check\ the\ event\ log\ to\ see\ all\ notifications=Controlla il registro eventi per vedere tutte le notifiche Host=Host Port=Porta @@ -1677,7 +1697,6 @@ Protected\ terms\ file=File di termini protetti Style\ file=File stile Open\ OpenOffice/LibreOffice\ connection=Apri connessione OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Devi inserire almeno il nome di un campo Non-ASCII\ encoded\ character\ found=Trovato un carattere non codificato ASCII Non-UTF-8\ encoded\ field\ found=Trovato campo codificato non-UTF-8 Toggle\ web\ search\ interface=Inverti la selezione dell'interfaccia di ricerca web @@ -1815,7 +1834,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Rimuove tutte l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Impossibile recuperare dati in ingresso da '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Non è stato possibile analizzare la voce %0. Invalid\ identifier\:\ '%0'.=Identificatore non valido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Questo documento è stato ritirato. empty\ citation\ key=chiave BibTeX vuota Aux\ file=File aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Gruppo contenente voci citate in un determinato file TeX @@ -2298,10 +2316,6 @@ Removes\ digits.=Rimuove le cifre. Presets=Profili -Check\ Proxy\ Setting=Controlla Impostazioni Proxy -Check\ connection=Controlla la connessione -Connection\ failed\!=Connessione fallita\! -Connection\ successful\!=Connessione riuscita\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Genera gruppi da parole chiave nel campo seguente Generate\ groups\ for\ author\ last\ names=Genera gruppi in base al cognome dell'autore Regular\ expression=Espressione regolare @@ -2359,7 +2373,6 @@ Entry\ Type=Tipo di voce Entry\ types=Tipi di voce Field\ names=Nome del campo Others=Altri -Overwrite\ existing\ field\ values=Sovrascrivi i valori esistenti del campo Recommended=Consigliato Authors\ and\ Title=Autori e Titolo @@ -2427,3 +2440,23 @@ Search\ results\ from\ open\ libraries=Risultati della ricerca da librerie apert Custom\ API\ key=Chiave API personalizzata Check\ %0\ API\ Key\ Setting=Controlla le impostazioni della chiave API %0 + +Edit\ content=Modifica il contenuto +Copy\ or\ Move\ content=Copia o Sposta il contenuto +Overwrite\ field\ content=Sovrascrivi il contenuto del campo +Set=Imposta +Append=Accoda +Clear\ field\ content=Cancella il contenuto del campo +Set\ or\ append\ content=Imposta o aggiungi il contenuto +Edit\ field\ content\ for\ selected\ entries=Modifica il contenuto del campo per le voci selezionate +Rename=Rinomina +New\ field\ name=Nuovo nome del campo +Copy\ content=Copia il contenuto +Move\ content=Sposta il contenuto +Swap\ content=Scambia il contenuto +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copia o sposta il contenuto di un campo in un altro +Automatic\ field\ editor=Editor automatico dei campi + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Nota\: se le voci originali mancano di parole chiave per qualificarsi per la nuova configurazione di gruppo, confermando qui le aggiungeranno) +Assign=Assegna +Do\ not\ assign=Non assegnare diff --git a/src/main/resources/l10n/JabRef_ja.properties b/src/main/resources/l10n/JabRef_ja.properties index d250ae84767..580a657d479 100644 --- a/src/main/resources/l10n/JabRef_ja.properties +++ b/src/main/resources/l10n/JabRef_ja.properties @@ -20,6 +20,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 項目 + %0\ matches\ the\ regular\ expression\ %1=%0は正規表現%1に一致します %0\ matches\ the\ term\ %1=%0は項目%1に一致します @@ -150,8 +151,6 @@ Cite\ command=Citeコマンド Clear=消去 -Clear\ fields=フィールドを消去 - Open\ /\ close\ entry\ editor=項目エディタを開閉する Close\ dialog=ダイアログを閉じる @@ -695,11 +694,18 @@ Remove\ selected\ entries\ from\ this\ group=このグループから選択項 Remove\ group=グループを削除 -Remove\ group\ "%0"?=グループ「%0」を削除しますか? +Remove\ group\ and\ subgroups=グループと下層グループを削除 + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=グループ「%0」とその下層グループを削除しますか? -Remove\ group\ and\ subgroups=グループと下層グループを削除 +Removed\ group\ "%0"\ and\ its\ subgroups.=グループ「%0」とその下層グループを削除しました. + + Remove\ link=リンクを削除 @@ -709,8 +715,6 @@ Remove\ string\ %0=文字列 %0 を削除 Removed\ group\ "%0".=グループ「%0」を削除しました. -Removed\ group\ "%0"\ and\ its\ subgroups.=グループ「%0」とその下層グループを削除しました. - Removed\ string=文字列を削除しました Renamed\ string=文字列を改名しました @@ -779,9 +783,6 @@ Select\ file\ from\ ZIP-archive=ZIP書庫からファイルを選択してくだ Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=ツリーノードを選択して表示させ,変更を受諾ないし拒否してください -Set\ field=フィールドを設定 -Set\ fields=フィールドを設定 - Settings=設定 Shortcut=ショートカット @@ -814,7 +815,6 @@ Size=サイズ Skipped\ -\ No\ PDF\ linked=跳ばしました - PDFがリンクされていません Skipped\ -\ PDF\ does\ not\ exist=跳ばしました - PDFが存在しません -Skipped\ entry.=項目を跳ばしました. Sort\ subgroups=サブグループのソート @@ -960,10 +960,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=%0行目\: No\ full\ text\ document\ found=文書本体が見つかりません Download\ from\ URL=URLからダウンロード Rename\ field=フィールドを改名しました -Append\ field=フィールドを追加 -Append\ to\ fields=フィールドに追加 -Rename\ field\ to=フィールド名を以下に変更 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=フィールドの内容を別名のフィールドに移動する Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=リモート操作用に%0ポートを使用することができません.別のアプリケーションが使用している可能性があります.別のポートを指定してみてください. @@ -1219,6 +1215,11 @@ Use\ custom\ proxy\ configuration=ユーザー定義のプロキシ設定を用 Proxy\ requires\ authentication=プロキシには認証が必要 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=警告:パスワードは平文で保管されます! Clear\ connection\ settings=接続設定を消去する +Check\ Proxy\ Setting=プロキシ設定を確認 +Check\ connection=接続を確認 +Connection\ failed\!=接続に失敗しました! +Connection\ successful\!=接続に成功しました ! + Open\ folder=フォルダを開く Export\ sort\ order=書き出し整序順 @@ -1668,7 +1669,6 @@ Protected\ terms\ file=予約語ファイル Style\ file=スタイルファイル Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice接続を開く -You\ must\ enter\ at\ least\ one\ field\ name=少なくとも一つのフィールド名を入力する必要があります Non-ASCII\ encoded\ character\ found=ASCIIエンコードでない文字が検出されました Non-UTF-8\ encoded\ field\ found=UTF-8でエンコードされていないフィールドが見つかりました Toggle\ web\ search\ interface=ウェブ検索インタフェースを入切 @@ -1806,7 +1806,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=フィールド Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' から項目データを取得することができませんでした. Entry\ from\ %0\ could\ not\ be\ parsed.=%0 からの項目を解析することができませんでした. Invalid\ identifier\:\ '%0'.=「%0」は識別子として無効です. -This\ paper\ has\ been\ withdrawn.=この論文は撤回されました. empty\ citation\ key=空の文献キー Aux\ file=auxファイル Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=特定のTeXファイルでの引用項目を集めたグループ @@ -2141,10 +2140,6 @@ Removes\ digits.=数字を削除します。 Presets=プリセット -Check\ Proxy\ Setting=プロキシ設定を確認 -Check\ connection=接続を確認 -Connection\ failed\!=接続に失敗しました! -Connection\ successful\!=接続に成功しました ! Generate\ groups\ from\ keywords\ in\ the\ following\ field=次のフィールドのキーワードからグループを生成する Generate\ groups\ for\ author\ last\ names=著者の姓でグループを生成する Regular\ expression=正規表現 @@ -2190,7 +2185,6 @@ Entry\ Type=項目型 Entry\ types=項目型 Field\ names=フィールド名\: Others=その他 -Overwrite\ existing\ field\ values=既存のフィールド値を上書き Recommended=推奨 Authors\ and\ Title=著者とタイトル @@ -2247,3 +2241,5 @@ Search\ results\ from\ open\ libraries=オープンライブラリからの検 + + diff --git a/src/main/resources/l10n/JabRef_ko.properties b/src/main/resources/l10n/JabRef_ko.properties index 2938d9c870e..90f720b7e6f 100644 --- a/src/main/resources/l10n/JabRef_ko.properties +++ b/src/main/resources/l10n/JabRef_ko.properties @@ -20,6 +20,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 항목 + %0\ matches\ the\ regular\ expression\ %1=%0은 정규식 %1와 일치합니다. %0\ matches\ the\ term\ %1=%0은 항목 %1 와 일치합니다. @@ -150,8 +151,6 @@ Cite\ command=인용 명령 Clear=초기화 -Clear\ fields=필드 초기화 - Open\ /\ close\ entry\ editor=항목 편집기 열기/닫기 Close\ dialog=대화상자 닫기 @@ -693,11 +692,18 @@ Remove\ selected\ entries\ from\ this\ group=이 그룹에서 선택한 항목 Remove\ group=그룹 삭제 -Remove\ group\ "%0"?=그룹 "%0"을 삭제하겠습니까? +Remove\ group\ and\ subgroups=그룹 및 하위 그룹 제거 + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=그룹 "%0" 및 해당 하위 그룹을 제거하시겠습니까? -Remove\ group\ and\ subgroups=그룹 및 하위 그룹 제거 +Removed\ group\ "%0"\ and\ its\ subgroups.=그룹 "%0" 및 해당 하위 그룹을 제거했습니다. + + Remove\ link=링크 삭제 @@ -707,8 +713,6 @@ Remove\ string\ %0=문자열 %0 삭제 Removed\ group\ "%0".=그룹 %0 삭제 -Removed\ group\ "%0"\ and\ its\ subgroups.=그룹 "%0" 및 해당 하위 그룹을 제거했습니다. - Removed\ string=문자열 삭제 완료 Renamed\ string=문자열 이름 변경 완료 @@ -776,9 +780,6 @@ Select\ file\ from\ ZIP-archive=ZIP 아카이브에서 파일 선택 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=변경 사항을 보고 수락하거나 거부할 트리 노드를 선택합니다. -Set\ field=필드 설정 -Set\ fields=필드 설정 - Settings=설정 Shortcut=단축키 @@ -811,7 +812,6 @@ Size=크기 Skipped\ -\ No\ PDF\ linked=건너뜀 - 연결된 PDF 없음 Skipped\ -\ PDF\ does\ not\ exist=건너뜀 - PDF가 존재하지 않습니다 -Skipped\ entry.=항목을 건너뛰었습니다. Sort\ subgroups=하위 그룹 정렬 @@ -955,10 +955,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=%0행\: 손 No\ full\ text\ document\ found=전체 텍스트 문서를 찾을 수 없습니다. Download\ from\ URL=URL에서 다운로드 Rename\ field=필드 이름 변경 -Append\ field=필드 추가 -Append\ to\ fields=필드에 추가 -Rename\ field\ to=필드 이름 변경 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=필드 내용을 다른 이름의 필드로 이동 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=원격 작업에 포트 %0을 사용할 수 없습니다. 다른 응용 프로그램에서 사용 중일 수 있습니다. 다른 포트를 지정해 보십시오. @@ -1214,6 +1210,11 @@ Use\ custom\ proxy\ configuration=사용자 지정 프록시 구성 사용 Proxy\ requires\ authentication=프록시에 인증이 필요합니다. Attention\:\ Password\ is\ stored\ in\ plain\ text\!=주의\: 비밀번호는 일반 텍스트로 저장됩니다\! Clear\ connection\ settings=연결 설정 지우기 +Check\ Proxy\ Setting=프록시 설정 확인 +Check\ connection=연결 확인 +Connection\ failed\!=연결 실패\! +Connection\ successful\!=연결 성공\! + Open\ folder=폴더 열기 Export\ sort\ order=정렬 순서 내보내기 @@ -1663,7 +1664,6 @@ Protected\ terms\ file=보호된 용어 파일 Style\ file=파일 스타일 Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice 연결 열기 -You\ must\ enter\ at\ least\ one\ field\ name=하나 이상의 필드 이름을 입력해야 합니다. Non-ASCII\ encoded\ character\ found=ASCII가 아닌 인코딩된 문자를 찾았습니다. Non-UTF-8\ encoded\ field\ found=UTF-8이 아닌 인코딩 필드를 찾았습니다. Toggle\ web\ search\ interface=웹 검색 인터페이스 전환 @@ -1801,7 +1801,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=필드 내용 Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0'에서 항목 데이터를 검색할 수 없습니다. Entry\ from\ %0\ could\ not\ be\ parsed.=%0의 항목을 구문 분석할 수 없습니다. Invalid\ identifier\:\ '%0'.=잘못된 식별자\: '%0'. -This\ paper\ has\ been\ withdrawn.=이 논문은 철회되었습니다. empty\ citation\ key=빈 인용 키 Aux\ file=AUX 파일 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=주어진 TeX 파일에서 인용된 항목을 포함하는 그룹 @@ -1905,7 +1904,7 @@ Add\ new\ String=문자열 추가 Must\ not\ be\ empty\!=비워둘 수 없습니다\! Open\ Help\ page=도움말 열기 Add\ new\ field\ name=새 필드 이름 추가 -Field\ name\:=필드 이름\: +Field\ name\:=필드 이름\: Field\ name\ "%0"\ already\ exists=필드 이름 "%0"이 이미 존재합니다 No\ field\ name\ selected\!=필드 이름을 선택하지 않았습니다 Remove\ field\ name=필드 이름 제거 @@ -2280,10 +2279,6 @@ Removes\ digits.=숫자 제거 Presets=프리셋 -Check\ Proxy\ Setting=프록시 설정 확인 -Check\ connection=연결 확인 -Connection\ failed\!=연결 실패\! -Connection\ successful\!=연결 성공\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=다음 필드의 키워드에서 그룹 생성 Generate\ groups\ for\ author\ last\ names=작가의 성씨를 가지고 그룹 생성 Regular\ expression=정규식 @@ -2340,7 +2335,6 @@ Entry\ Type=목록 유형 Entry\ types=목록 유형들 Field\ names=필드 이름 Others=기타 -Overwrite\ existing\ field\ values=기존의 필드 값 덮어쓰기 Recommended=권장 Authors\ and\ Title=작가와 제목 @@ -2401,3 +2395,5 @@ Keep\ dialog\ always\ on\ top=대화창 위로 고정 + + diff --git a/src/main/resources/l10n/JabRef_nl.properties b/src/main/resources/l10n/JabRef_nl.properties index 6b183bbe7f9..312e3c7e358 100644 --- a/src/main/resources/l10n/JabRef_nl.properties +++ b/src/main/resources/l10n/JabRef_nl.properties @@ -1,3 +1,4 @@ +Could\ not\ delete\ empty\ entries.=Kan lege invoer niet verwijderen. @@ -13,6 +14,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 komt overeen met de standaard-uitdruk %1 %0\ matches\ the\ term\ %1=%0 komt overeen met de term %1 @@ -135,8 +137,6 @@ Cite\ command=Citeer opdracht Clear=Wissen -Clear\ fields=Velden wissen - Close\ dialog=Sluit dialoog @@ -638,11 +638,18 @@ Remove\ selected\ entries\ from\ this\ group=Geselecteerde items uit deze groep Remove\ group=Verwijder groep -Remove\ group\ "%0"?=Verwijder groep "%0"? +Remove\ group\ and\ subgroups=Verwijder groep en subgroepen + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Verwijder groep "%0" en zijn subgroepen? -Remove\ group\ and\ subgroups=Verwijder groep en subgroepen +Removed\ group\ "%0"\ and\ its\ subgroups.=Groep "%0" en zijn subgroepen verwijderd. + + Remove\ link=Verwijder link @@ -651,8 +658,6 @@ Remove\ old\ entry=Verwijder oude invoer Removed\ group\ "%0".=Groep "%0" verwijderd. -Removed\ group\ "%0"\ and\ its\ subgroups.=Groep "%0" en zijn subgroepen verwijderd. - Removed\ string=Tekenreeks verwijderd Renamed\ string=Tekenreeks hernoemd @@ -719,9 +724,6 @@ Select\ file\ from\ ZIP-archive=Selecteer bestand van ZIP-archief Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecteer de boom knopen om veranderingen te tonen en te accepteren of afwijzen -Set\ field=Veld instellen -Set\ fields=Velden instellen - Settings=Instellingen Shortcut=Snelkoppeling @@ -754,7 +756,6 @@ Size=Grootte Skipped\ -\ No\ PDF\ linked=Overgeslagen - Geen PDF gelinkt Skipped\ -\ PDF\ does\ not\ exist=Overgeslagen - PDF bestaat niet -Skipped\ entry.=Overgeslagen invoer. source\ edit=broncode aanpassen @@ -872,10 +873,6 @@ MIME\ type=MIME type No\ full\ text\ document\ found=Geen volledig tekstdocument gevonden Download\ from\ URL=Download vanaf URL Rename\ field=Veld hernoemen -Append\ field=Veld toevoegen -Append\ to\ fields=Toevoegen aan velden -Rename\ field\ to=Veld hernoemen naar -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Verplaats de inhoud van een veld naar een veld met een andere naam Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan poort %0 niet gebruiken voor bediening op afstand; een andere applicatie heeft deze mogelijk in gebruik. Probeer een andere poort op te geven. @@ -1090,6 +1087,7 @@ Proxy\ requires\ authentication=Proxy vereist authenticatie Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Opgelet\: Wachtwoord wordt opgeslagen in platte tekst\! Clear\ connection\ settings=Verbindingsinstellingen wissen + Open\ folder=Map openen Export\ sort\ order=Sorteervolgorde exporteren @@ -1494,7 +1492,6 @@ Protected\ terms\ file=Beschermde voorwaarden bestand Style\ file=Bestandsstijl Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice verbinding openen -You\ must\ enter\ at\ least\ one\ field\ name=U moet ten minste één veldnaam opgeven Non-ASCII\ encoded\ character\ found=Niet-ASCII gecodeerd teken gevonden Toggle\ web\ search\ interface=Web zoek interface in-/uitschakelen @@ -1615,7 +1612,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Verwijdert alle Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Kon geen data ophalen van '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Invoer van %0 kon niet worden geparseerd. Invalid\ identifier\:\ '%0'.=Ongeldig Id\: '%0'. -This\ paper\ has\ been\ withdrawn.=Dit artikel is ingetrokken. Aux\ file=Aux bestand Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Groep bevat invoergegevens aangehaald in een bepaald TeX-bestand @@ -1763,6 +1759,8 @@ Default\ pattern=Standaard patroon + + diff --git a/src/main/resources/l10n/JabRef_no.properties b/src/main/resources/l10n/JabRef_no.properties index 164dcbe8bb2..d8de347be33 100644 --- a/src/main/resources/l10n/JabRef_no.properties +++ b/src/main/resources/l10n/JabRef_no.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 matcher regulæruttrykket %1 %0\ matches\ the\ term\ %1=%0 matcher uttrykket %1 @@ -126,8 +127,6 @@ Cite\ command=Siteringskommando Clear=Opphev -Clear\ fields=Slett felter - Close\ dialog=Lukk dialog @@ -604,11 +603,18 @@ Remove\ selected\ entries\ from\ this\ group=Fjern valgte oppføringer fra denne Remove\ group=Fjern gruppe -Remove\ group\ "%0"?=Fjern gruppen "%0"? +Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Fjern gruppen "%0" og dens undergrupper? -Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernet gruppen "%0" og dens undergrupper + + Remove\ link=Slett link @@ -617,8 +623,6 @@ Remove\ old\ entry=Fjern gammel enhet Removed\ group\ "%0".=Fjernet gruppen "%0" -Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernet gruppen "%0" og dens undergrupper - Removed\ string=Fjernet streng Renamed\ string=Endret navn på streng @@ -681,9 +685,6 @@ Select\ file\ from\ ZIP-archive=Velg fil fra ZIP-fil Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Velg trenodene for å inspisere og akseptere eller avslå endringer -Set\ field=Sett felt -Set\ fields=Sett felter - Settings=Innstillinger Shortcut=Snarvei @@ -714,7 +715,6 @@ Size=Størrelse Skipped\ -\ No\ PDF\ linked=Hoppet over - ingen PDF-fil linket Skipped\ -\ PDF\ does\ not\ exist=Hoppet over - PDF-filen finnes ikke -Skipped\ entry.=Hoppet over enhet. source\ edit=redigering av kilde @@ -828,8 +828,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Last ned fra URL Rename\ field=Endre navn på felt -Rename\ field\ to=Endre navn på felt til -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytt innholdet i et felt til et annet felt Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan ikke bruke port %0 for fjernstyring; den kan være i bruk av et annet program. Pr¸v å spesifisere en annen port. @@ -962,6 +960,7 @@ Finished\ automatically\ setting\ external\ links.=Fullførte automatisk setting + Open\ folder=Åpne mapp Export\ sort\ order=Sorteringsrekkefølge ved eksport @@ -1203,6 +1202,8 @@ Default\ pattern=Standardmønster + + diff --git a/src/main/resources/l10n/JabRef_pl.properties b/src/main/resources/l10n/JabRef_pl.properties index 153fa9456a9..e8a82418acc 100644 --- a/src/main/resources/l10n/JabRef_pl.properties +++ b/src/main/resources/l10n/JabRef_pl.properties @@ -13,6 +13,7 @@ + Abbreviation=Skrót About\ JabRef=O JabRef @@ -95,8 +96,6 @@ Change\ entry\ type=Zmień typ wpisu Clear=Wyczyść -Clear\ fields=Wyczyść pola - Close\ dialog=Zamknij okno @@ -472,9 +471,16 @@ Remove\ subgroups=Usuń podgrupy Remove\ group=Usuń grupę +Remove\ group\ and\ subgroups=Usuń grupę i podgrupy + + + + + + + -Remove\ group\ and\ subgroups=Usuń grupę i podgrupy Remove\ link=Usuń łącze @@ -484,7 +490,6 @@ Remove\ old\ entry=Usuń stary wpis - Replace=Zamień Replace\ With\:=Zamień na\: Find\:=Znajdź\: @@ -529,7 +534,6 @@ Select\ entry\ type=Wybierz typ wpisu - Settings=Ustawienia Shortcut=Skrót @@ -551,7 +555,6 @@ Simple\ HTML=Uproszczony HTML Size=Rozmiar -Skipped\ entry.=Pominięty wpis. @@ -627,7 +630,6 @@ MIME\ type=Typ MIME Download\ from\ URL=Pobierz z adresu URL Rename\ field=Zmień nazwę pola -Append\ to\ fields=Dołącz do pól Please\ enter\ a\ valid\ file\ path.=Wprowadź prawidłową ścieżkę do pliku. @@ -717,6 +719,7 @@ Hostname=Nazwa hosta Proxy\ configuration=Konfiguracja proxy + Open\ folder=Otwórz folder Parsing\ error=Błąd parsowania @@ -1077,6 +1080,8 @@ Reset=Reset + + diff --git a/src/main/resources/l10n/JabRef_pt.properties b/src/main/resources/l10n/JabRef_pt.properties index a4cf86dab14..6fd581ffae8 100644 --- a/src/main/resources/l10n/JabRef_pt.properties +++ b/src/main/resources/l10n/JabRef_pt.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 encontrada a expressão regular %1 %0\ matches\ the\ term\ %1=%0 encontrado o termo %1 @@ -130,8 +131,6 @@ Cite\ command=Comando citar Clear=Limpar -Clear\ fields=Limpar campos - Close\ dialog=Fechar janela de diálogo @@ -629,11 +628,18 @@ Remove\ entry\ from\ import=Remover referência da importação Remove\ group=Remover grupo -Remove\ group\ "%0"?=Remover grupo "%0"? +Remove\ group\ and\ subgroups=Remover grupos e subgrupos + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Remover grupo "%0" e seus subgrupos? -Remove\ group\ and\ subgroups=Remover grupos e subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. + + Remove\ link=Remover link @@ -642,8 +648,6 @@ Remove\ old\ entry=Remover referência antiga Removed\ group\ "%0".=Grupo "%0" removido. -Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. - Removed\ string=String removida Renamed\ string=String renomeada @@ -696,9 +700,6 @@ Select\ file\ from\ ZIP-archive=Selecionar arquivo a partir de um arquivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecione os nós da árvore para visualizar e aceitar ou rejeitar mudanças -Set\ field=Configurar campo -Set\ fields=Configurar campos - Settings=Configurações Shortcut=Atalho @@ -729,7 +730,6 @@ Size=Tamanho Skipped\ -\ No\ PDF\ linked=Omitido - Nenhum PDF linkado Skipped\ -\ PDF\ does\ not\ exist=Omitido - O PDF não existe -Skipped\ entry.=Referência omitida. source\ edit=edição de fonte @@ -840,8 +840,6 @@ Unable\ to\ open\ link.=Não foi possível abrir link. This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ running\ instance\ of\ JabRef\ instead\ of\ opening\ a\ new\ instance.\ For\ instance,\ this\ is\ useful\ when\ you\ open\ a\ file\ in\ JabRef\ from\ your\ web\ browser.\ Note\ that\ this\ will\ prevent\ you\ from\ running\ more\ than\ one\ instance\ of\ JabRef\ at\ a\ time.=Esta funcionalidade permite que novos arquivos sejam abertos ou importados para uma instância do JabRef já aberta ao invés de abrir uma nova instância. Por exemplo, isto é útil quando você abre um arquivo no JabRef a partir de ser navegador web. Note que isto irá previnir que você execute uma ou mais instâncias do JabRef ao mesmo tempo. Rename\ field=Renomear campo -Rename\ field\ to=Renomear campo para -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover conteúdo de um campo para um campo com nome diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Não é possível utilizar a porta %0 para operação remota; outra aplicação pode estar usando-a. Tente utilizar uma outra porta. @@ -1050,6 +1048,7 @@ Use\ custom\ proxy\ configuration=Usar configurações personalizadas de proxy Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atenção\: A senha é armazenada em texto simples\! Clear\ connection\ settings=Limpar configurações da conexão + Open\ folder=Abrir diretório Export\ sort\ order=Exportar ordenação @@ -1355,6 +1354,8 @@ Default\ pattern=Ppadrão predefinido + + diff --git a/src/main/resources/l10n/JabRef_pt_BR.properties b/src/main/resources/l10n/JabRef_pt_BR.properties index f75024aafc7..179f7dd8128 100644 --- a/src/main/resources/l10n/JabRef_pt_BR.properties +++ b/src/main/resources/l10n/JabRef_pt_BR.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entradas -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 exportação bem-sucedida. Abrir a pasta que contém o arquivo salvo? + %0\ matches\ the\ regular\ expression\ %1=%0 encontrada a expressão regular %1 @@ -151,8 +151,6 @@ Cite\ command=Comando citar Clear=Limpar -Clear\ fields=Limpar campos - Open\ /\ close\ entry\ editor=Abrir / fechar o editor de referências Close\ dialog=Fechar janela de diálogo @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Remover as referências selecionada Remove\ group=Remover grupo -Remove\ group\ "%0"?=Remover grupo "%0"? +Remove\ group\ and\ subgroups=Remover grupos e subgrupos + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Remover grupo "%0" e seus subgrupos? -Remove\ group\ and\ subgroups=Remover grupos e subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. + + Remove\ link=Remover link @@ -717,8 +722,6 @@ Remove\ string\ %0=Remover string %0 Removed\ group\ "%0".=Grupo "%0" removido. -Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. - Removed\ string=String removida Renamed\ string=String renomeada @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=Selecionar arquivo a partir de um arquivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecione os nós da árvore para visualizar e aceitar ou rejeitar mudanças -Set\ field=Configurar campo -Set\ fields=Configurar campos - Settings=Configurações Shortcut=Atalho @@ -825,7 +825,6 @@ Size=Tamanho Skipped\ -\ No\ PDF\ linked=Omitido - Nenhum PDF linkado Skipped\ -\ PDF\ does\ not\ exist=Omitido - O PDF não existe -Skipped\ entry.=Referência omitida. Sort\ subgroups=Ordenar subgrupos @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Linha %0\: En No\ full\ text\ document\ found=Nenhum documento encontrado Download\ from\ URL=Baixar da URL Rename\ field=Renomear campo -Append\ field=Acrescentar campo -Append\ to\ fields=Acrescentar aos campos -Rename\ field\ to=Renomear campo para -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover conteúdo de um campo para um campo com nome diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Não é possível utilizar a porta %0 para operação remota; outra aplicação pode estar usando-a. Tente utilizar uma outra porta. @@ -1242,6 +1237,11 @@ Use\ custom\ proxy\ configuration=Usar configurações personalizadas de proxy Proxy\ requires\ authentication=Proxy requer autenticação Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atenção\: A senha é armazenada em texto simples\! Clear\ connection\ settings=Limpar configurações da conexão +Check\ Proxy\ Setting=Verificar Configuração de Proxy +Check\ connection=Verificar conexão +Connection\ failed\!=Falha na conexão\! +Connection\ successful\!=Conexão bem-sucedida\! + Open\ folder=Abrir diretório Export\ sort\ order=Exportar ordenação @@ -1694,7 +1694,6 @@ Protected\ terms\ file=Arquivo de termos protegidos Style\ file=Arquivo de estilo Open\ OpenOffice/LibreOffice\ connection=Abrir conexão com OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Você deve digitar pelo menos um nome de campo Non-ASCII\ encoded\ character\ found=Caractere não-ASCII encontrado Non-UTF-8\ encoded\ field\ found=Campo não-UTF-8 encontrado Toggle\ web\ search\ interface=Alternar interface de busca na web @@ -1832,7 +1831,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Remove todas as Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Não foi possível recuperar dados de referência a partir de '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Não foi possível analisar a referência a partir de %0. Invalid\ identifier\:\ '%0'.=Identificador inválido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Este documento foi retirado. empty\ citation\ key=chave de citação vazia Aux\ file=Arquivo Aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo contendo referências citadas em um arquivo TeX @@ -2314,10 +2312,6 @@ Removes\ digits.=Remove dígitos. Presets=Predefinições -Check\ Proxy\ Setting=Verificar Configuração de Proxy -Check\ connection=Verificar conexão -Connection\ failed\!=Falha na conexão\! -Connection\ successful\!=Conexão bem-sucedida\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Gerar grupos de palavras-chave no campo seguinte Generate\ groups\ for\ author\ last\ names=Gerar grupos para os últimos nomes dos autores Regular\ expression=Expressão regular @@ -2375,7 +2369,6 @@ Entry\ Type=Tipo de referência Entry\ types=Tipos de referência Field\ names=Nomes dos campos Others=Outros -Overwrite\ existing\ field\ values=Sobrescrever valores de campos existentes Recommended=Recomendado Authors\ and\ Title=Autores e Título @@ -2439,3 +2432,5 @@ Search\ results\ from\ open\ libraries=Resultados da pesquisa para as biblioteca + + diff --git a/src/main/resources/l10n/JabRef_ru.properties b/src/main/resources/l10n/JabRef_ru.properties index 4f2f56445c7..0999c1ddcb0 100644 --- a/src/main/resources/l10n/JabRef_ru.properties +++ b/src/main/resources/l10n/JabRef_ru.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 записей +Export\ operation\ finished\ successfully.=Операция экспорта выполнена успешно. + +Reveal\ in\ File\ Explorer=Открыть в Проводнике %0\ matches\ the\ regular\ expression\ %1=%0 соответствует регулярному выражению %1 @@ -150,8 +153,6 @@ Cite\ command=Команда цитирования Clear=Очистить -Clear\ fields=Очистить поля - Open\ /\ close\ entry\ editor=Открыть / закрыть редактор записей Close\ dialog=Закрыть диалоговое окно @@ -173,6 +174,8 @@ Copy=Копировать Copy\ title=Копировать заглавие Copy\ \\cite{citation\ key}=Копировать \\ цитировать{citation key} +Copy\ citation\ (html)=Копировать цитирование (html) +Copy\ citation\ (text)=Копировать цитирование (текст) Copy\ citation\ key=Копировать ключ цитаты Copy\ citation\ key\ and\ link=Скопировать ключ цитаты и ссылку Copy\ citation\ key\ and\ title=Копировать ключ цитаты и заголовок @@ -230,6 +233,8 @@ Library\ properties=Свойства БД Default=По умолчанию +Character\ encoding\ UTF-8\ is\ not\ supported.=Кодировка UTF-8 не поддерживается. +UTF-8\ could\ not\ be\ used\ to\ encode\ the\ following\ characters\:\ %0=Не удалось использовать UTF-8 для кодирования следующих символов\: %0 The\ chosen\ encoding\ '%0'\ could\ not\ encode\ the\ following\ characters\:=В выбранной кодировке '%0' невозможно прочитать следующие символы\: Downloading=Загрузка … @@ -278,6 +283,7 @@ Downloaded\ website\ as\ an\ HTML\ file.=Загружен сайт в форма duplicate\ removal=удаление дубликатов +Duplicate\ fields=Дубликаты полей Duplicate\ string\ name=Дубликат имени строки @@ -536,7 +542,9 @@ Moved\ group\ "%0".=Перемещенная группа "%0". Mr.\ DLib\ Privacy\ settings=Настройки конфиденциальности сервиса "Mr. DLib" +No\ database\ is\ open=База данных не открыта +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=Нам требуется база данных для экспорта. Откройте её. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=Веб сервис "Mr. DLib" не предлагает рекомендаций для этой записи. @@ -695,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Удалить выбранные Remove\ group=Удалить группу -Remove\ group\ "%0"?=Удалить группу "%0"? +Remove\ group\ and\ subgroups=Удалить группу и подгруппы + +Remove\ groups\ and\ subgroups=Удалить группы и подгруппы + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Удалить все выбранные группы, сохранив их подгруппы? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Удалить группу "%0", сохранив её подгруппы? + +Remove\ groups=Удалить группы + +Removed\ all\ selected\ groups.=Все выбранные группы удалены. Remove\ group\ "%0"\ and\ its\ subgroups?=Удалить группу "%0" и ее подгруппы? -Remove\ group\ and\ subgroups=Удалить группу и подгруппы +Removed\ group\ "%0"\ and\ its\ subgroups.=Группа "%0" и ее подгруппы удалены. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Удалить все выбранные группы и их подгруппы? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Все выбранные группы и их подгруппы удалены. Remove\ link=Удалить ссылку @@ -709,8 +731,6 @@ Remove\ string\ %0=Удалить строку %0 Removed\ group\ "%0".=Группа "%0" удалена. -Removed\ group\ "%0"\ and\ its\ subgroups.=Группа "%0" и ее подгруппы удалены. - Removed\ string=Строка удалена Renamed\ string=Строка переименована @@ -733,6 +753,8 @@ Replaces\ Unicode\ ligatures\ with\ their\ expanded\ form=Заменяет ли Required\ fields=Обязательные поля +Do\ not\ resolve\ BibTeX\ strings=Не разрешать строки BibTeX +Resolve\ BibTeX\ strings\ for\ the\ following\ fields=Разрешить строки BibTeX для следующих полей resolved=разрешено @@ -740,11 +762,13 @@ Restart=Перезапустить Restart\ required=Требуется перезапуск +Return\ to\ dialog=Вернуться в диалоговое окно Review=Просмотр Review\ changes=Просмотр изменений Review\ Field\ Migration=Перемещение поля обзора +Loading=Загрузка Save=Сохранить Save\ all\ finished.=Сохранить все завершенные. @@ -778,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Выбрать файл из ZIP-архива Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Выбрать узлы дерева для просмотра и применения/отклонения изменений -Set\ field=Настройка поля -Set\ fields=Настройка полей - Settings=Параметры Shortcut=Ярлык @@ -813,7 +834,12 @@ Size=Размер Skipped\ -\ No\ PDF\ linked=Пропущено - Без ссылок на PDF Skipped\ -\ PDF\ does\ not\ exist=Пропущено - PDF не существует -Skipped\ entry.=Запись пропущена. +JabRef\ skipped\ the\ entry.=JabRef пропустил запись. +Import\ error=Ошибка импортирования +Open\ library\ error=Ошибка при открытии библиотеки +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Пожалуйста, проверьте ваш файл библиотеки на правильность синтаксиса. +SourceTab\ error=Ошибка SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Пользовательский ввод посредством редактора записей во вкладке `{}bibtex source` привёл к возникновению ошибки. Sort\ subgroups=Сортировать подгруппы @@ -902,6 +928,7 @@ Warning=Предупреждение Warnings=Предупреждения +Warning\:\ You\ added\ field\ "%0"\ twice.\ Only\ one\ will\ be\ kept.=Внимание\: Вы добавили поле "%0" дважды. Только одно из них будет сохранено. web\ link=веб-ссылки @@ -959,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Строка No\ full\ text\ document\ found=Не найдено полнотекстовых документов Download\ from\ URL=Загрузить из URL-адреса Rename\ field=Переименовать поле -Append\ field=Добавить поле -Append\ to\ fields=Добавить к полям -Rename\ field\ to=Переименовать поле в -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Переместить содержимое поля в поле с другим именем Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Невозможно использовать порт %0 для удаленного подключения; another application may be using it. Try specifying another port. @@ -1032,6 +1055,7 @@ Cite\ special=Особое цитирование Extra\ information\ (e.g.\ page\ number)=Дополнительные сведения (напр., номер страницы) Manage\ citations=Управление цитированием Problem\ modifying\ citation=Ошибка при модификации цитаты +Problem\ collecting\ citations=Возникла проблема при сборе ссылок Citation=Цитата Connecting...=Подключение... Select\ style=Выбор стиля @@ -1046,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=Синхронизация библи Select\ which\ open\ Writer\ document\ to\ work\ on=Выберите открытый документ OpenOffice/LibreOffice Writer для работы Connected\ to\ document=Подключено к документу +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Не удалось выполнить подключение к документу Writer. Убедитесь, что документ открыт перед использованием кнопки 'Выбрать документ Writer', чтобы выполнить подключение. +No\ Writer\ documents\ found=Документы Writer не найдены Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=Вставка цитаты без текста (запись отображается в списке ссылок) Cite\ selected\ entries\ with\ extra\ information=Цитировать выбранные записи с дополнительными сведениями @@ -1054,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=Проверьте актуал Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=Ваш документ OpenOffice/LibreOffice ссылается на ключ цитирования '%0', который не найден в вашей текущей библиотеке. +This\ operation\ requires\ a\ bibliography\ database.=Этой операции требуется библиографическая база данных. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Ссылки в вашем документе OpenOffice/LibreOffice содержат не менее %0 ключей цитирования, которые не удалось найти в текущей библиотеке. Некоторые из них %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=В вашем документе OpenOffice/LibreOffice отсутствуют какие-либо ключи цитирования, которые также можно найти в вашей текущей библиотеке. Unable\ to\ synchronize\ bibliography=Не удалось выполнить синхронизацию библиографии Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=Объединить пары цитат, разделенные только пробелом @@ -1063,7 +1092,12 @@ Autodetection\ failed=Ошибка автоопределения Please\ wait...=Подождите... Connection\ lost=Соединение прервано +Could\ not\ update\ bibliography=Не удалось обновить библиографию +Not\ connected\ to\ document=Нет подключения к документу +Problem\ combining\ cite\ markers=Проблема при объединении отметок цитирования +Unable\ to\ reload\ style\ file=Не удалось перезагрузить файл оформления +Problem\ during\ separating\ cite\ markers=Проблема при разделении отметок цитирования Automatically\ sync\ bibliography\ when\ inserting\ citations=Автоматически синхронизировать библиографию при вставке цитат Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=Поиск записей BibTeX только в активной вкладке @@ -1083,6 +1117,7 @@ Style\ selection=Выбор стиля No\ valid\ style\ file\ defined=Не определен допустимый файл стиля Choose\ pattern=Выбор шаблона Search\ and\ store\ files\ relative\ to\ library\ file\ location=Поиск и хранение файлов относительно файла библиотеки +File\ directory=Каталог файлов Could\ not\ run\ the\ gnuclient/emacsclient\ program.\ Make\ sure\ you\ have\ the\ emacsclient/gnuclient\ program\ installed\ and\ available\ in\ the\ PATH.=Не удалось запустить приложение gnuclient/emacsclient. Убедитесь, что приложение gnuclient/emacsclient установлено и доступно по указанному пути PATH. You\ must\ select\ either\ a\ valid\ style\ file,\ or\ use\ one\ of\ the\ default\ styles.=Необходимо использовать допустимый файл стиля или один из стилей по умолчанию. @@ -1165,6 +1200,7 @@ Toggle\ relevance=Показать/скрыть релевантность Toggle\ quality\ assured=Прказать/скрыть проверку контроля качества Toggle\ print\ status=Изменить статус 'печать' Update\ keywords=Изменить ключевые слова +Problem\ connecting=Проблема при подключении Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=Соединение с OpenOffice/LibreOffice прервано. Убедитесь, что приложение OpenOffice/LibreOffice запущено и повторите соединение. JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef отправляет по крайней мере один запрос для заметки издателю. @@ -1179,6 +1215,7 @@ Removed\ all\ subgroups\ of\ group\ "%0".=Все подгруппы группы To\ disable\ the\ memory\ stick\ mode\ rename\ or\ remove\ the\ jabref.xml\ file\ in\ the\ same\ folder\ as\ JabRef.=Для отключения режима флеш-памяти переименуйте или удалите файл jabref.xml в каталоге JabRef. Unable\ to\ connect.\ One\ possible\ reason\ is\ that\ JabRef\ and\ OpenOffice/LibreOffice\ are\ not\ both\ running\ in\ either\ 32\ bit\ mode\ or\ 64\ bit\ mode.=Не удалось выполнить подключение. Возможная причина\: JabRef и OpenOffice/LibreOffice не запущены в режиме 32-х или 64-х битного приложения. Delimiter(s)=Разделитель(и) +When\ downloading\ files,\ or\ moving\ linked\ files\ to\ the\ file\ directory,\ use\ the\ bib\ file\ location.=При загрузке файлов или перемещении связанных файлов в каталог файлов используйте расположение файла bib. Searching...=Выполняется поиск... Please\ enter\ a\ search\ string=Введите строку для поиска @@ -1216,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Использовать пользовате Proxy\ requires\ authentication=Для прокси-сервера требуется авторизация Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Внимание. Пароль сохранен в виде простого текста. Clear\ connection\ settings=Удалить настройки подключения +Check\ Proxy\ Setting=Проверить настройки прокси +Check\ connection=Проверить подключение +Connection\ failed\!=Соединение потеряно\! +Connection\ successful\!=Соединение успешно\! + +SSL\ Configuration=Конфигурация SSL +SSL\ configuration\ changed=Конфигурация SSL изменена +SSL\ certificate\ file=Файл сертификата SSL +Duplicate\ Certificates=Дубликаты сертификатов +You\ already\ added\ this\ certificate=Вы уже добавили этот сертификат Open\ folder=Открыть папку Export\ sort\ order=Экспорт в порядке сортировки @@ -1252,6 +1299,7 @@ Please\ open\ %0\ manually.=Откройте %0 вручную. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Ссылка скопирована в буфер обмена. Open\ %0\ file=Открыть файл %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Не удалось обнаружить терминал автоматически. Пожалуйста, укажите пользовательский терминал в настройках. Cannot\ delete\ file=Невозможно удалить файл File\ permission\ error=Ошибка прав доступа для файла @@ -1558,10 +1606,21 @@ Custom=Определено пользователем Export\ cited=Экспорт цитированного Unable\ to\ generate\ new\ library=Не удалось создать новую БД +The\ cursor\ is\ in\ a\ protected\ area.=Курсор находится в защищённой области. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=Не открыто библиографических баз данных для цитирования. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=Не открыто библиографических баз данных для обновления отметок ссылок после цитирования. +No\ bibliography\ entries\ are\ selected\ for\ citation.=Не выбраны библиографические записи для выполнения цитирования. +No\ bibliography\ style\ is\ selected\ for\ citation.=Не выбрано оформление библиографии для выполнения цитирования. +No\ database=Нет базы данных +No\ entries\ selected\ for\ citation=Не выбрано записей для выполнения цитирования +Open\ one\ before\ citing.=Откройте одну из них перед цитированием. +Select\ one\ before\ citing.=Выберите одну из них перед цитированием. +Select\ some\ before\ citing.=Выберите несколько перед выполнением цитирования. Found\ identical\ ranges=Найдены идентичные диапазоны Found\ overlapping\ ranges=Найдены пересекающиеся диапазоны @@ -1628,6 +1687,8 @@ Issue\ report\ successful=Отчёт об ошибке успешно отпра Your\ issue\ was\ reported\ in\ your\ browser.=Отчет о проблеме был отправлен в браузер. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Лог и информация об ошибке успешно скопированы. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Пожалуйста, вставьте эту информацию (с помощью Ctrl+V) в описание проблемы. +Last\ notification=Последнее уведомление +Check\ the\ event\ log\ to\ see\ all\ notifications=Проверьте журнал событий для просмотра всех уведомлений Host=Хост Port=Порт @@ -1665,7 +1726,6 @@ Protected\ terms\ file=Файл с защищенными терминами Style\ file=Файл стиля Open\ OpenOffice/LibreOffice\ connection=Установить подключение к Open OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Необходимо ввести минимум одно имя поля Non-ASCII\ encoded\ character\ found=Обнаружен символ не в кодировке ASCII Non-UTF-8\ encoded\ field\ found=Найдено поле НЕ в кодировке UTF-8 Toggle\ web\ search\ interface=Переключение интерфейса веб-поиска @@ -1803,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Удаляет Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Не удалось получить данные из записи '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Данные в %0 не могут быть обработаны. Invalid\ identifier\:\ '%0'.=Неверный идентификатор\: '%0'. -This\ paper\ has\ been\ withdrawn.=Настоящий документ был отозван. empty\ citation\ key=пустой ключ цитаты Aux\ file=Aux файл Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Группа, содержащая записи в заданном TeX файле @@ -1831,6 +1890,7 @@ Could\ not\ connect\ to\ Vim\ server.\ Make\ sure\ that\ Vim\ is\ running\ with\ Could\ not\ connect\ to\ a\ running\ gnuserv\ process.\ Make\ sure\ that\ Emacs\ or\ XEmacs\ is\ running,\ and\ that\ the\ server\ has\ been\ started\ (by\ running\ the\ command\ 'server-start'/'gnuserv-start').=Не удалось подключиться к запущенному процессу gnuserv. Убедитесь, что приложение Emacs или XEmacsи сервер запущены(с помощью команды 'server-start'/'gnuserv-start'). Error\ pushing\ entries=Ошибка при отправке записей +Preamble=Преамбула Markings=Маркировка Use\ selected\ instance=Использовать выбранный экземпляр @@ -1849,6 +1909,7 @@ Blog=Блог Check\ integrity=Проверка целостности Cleanup\ URL\ link=Очистить URL ссылку Cleanup\ URL\ link\ by\ removing\ special\ symbols\ and\ extracting\ simple\ link=Очистка URL ссылку, удалив специальные символы, чтобы получился простой URL +Copy\ DOI=Копировать DOI Copy\ DOI\ url=Копировать URL-адрес DOI Development\ version=Версия для разработчиков Export\ selected\ entries=Экспорт выбранных записей @@ -1858,6 +1919,8 @@ JabRef\ resources=Ресурсы JabRef Manage\ journal\ abbreviations=Настроить аббревиатуры журналов Manage\ protected\ terms=Управление защищенными терминами New\ entry\ from\ plain\ text=Новая запись из обычного текста +Import\ by\ ID=Импортировать по ID +Enter\ a\ valid\ ID=Введите действительный ID New\ sublibrary\ based\ on\ AUX\ file=Новая под-библиотека из файла AUX Push\ entries\ to\ external\ application\ (%0)=Передать записи во внешнее приложение (%0) Quit=Выход @@ -1917,6 +1980,7 @@ Keyword\ separator=Разделитель ключевых слов Remove\ keyword=Удалить слово Are\ you\ sure\ you\ want\ to\ remove\ keyword\:\ "%0"?=Вы уверены, что вы хотите удалить слово\: "%0"? Reset\ to\ default=Сбросить настройки +String\ constants=Строковые постоянные Export\ all\ entries=Экспорт всех записей Generate\ citation\ keys=Генерировать ключи цитирования Manage\ field\ names\ &\ content=Управлять названиями полей и контентом @@ -2046,6 +2110,7 @@ Please\ provide\ a\ valid\ aux\ file.=Пожалуйста, предоставь Keyword\ delimiter=Разделитель ключевых слов Hierarchical\ keyword\ delimiter=Разделитель иерархических ключевых слов Escape\ ampersands=Экранировка амперсандов +Escape\ dollar\ sign=Экранированный символ доллара Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Подсказка\:\nдля поиска всех полей со значением Smith, введите\:\nsmith\nДля поиска в поле Автор значения Smith а в поле Заголовок значения electrical, введите\:\nauthor\=smith and title\=electrical @@ -2234,7 +2299,7 @@ This\ entry\ type\ is\ intended\ for\ sources\ such\ as\ web\ sites\ which\ are\ A\ single-volume\ work\ of\ reference\ such\ as\ an\ encyclopedia\ or\ a\ dictionary.=Неделимая работа или ссылка, как энциклопедия или словарь. A\ technical\ report,\ research\ report,\ or\ white\ paper\ published\ by\ a\ university\ or\ some\ other\ institution.=Технический отчет, исследовательский отчет, или белая книга, выпущенная институтом или другим учреждением. An\ entry\ set\ is\ a\ group\ of\ entries\ which\ are\ cited\ as\ a\ single\ reference\ and\ listed\ as\ a\ single\ item\ in\ the\ bibliography.=Набор записей представляет собой группу записей, которые приводятся в виде единой ссылки и перечислены в виде одного элемента в библиографии. -Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. +Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. Supplemental\ material\ in\ a\ "Collection".=Дополнительные материалы в "Коллекции". Supplemental\ material\ in\ a\ "Periodical".\ This\ type\ may\ be\ useful\ when\ referring\ to\ items\ such\ as\ regular\ columns,\ obituaries,\ letters\ to\ the\ editor,\ etc.\ which\ only\ have\ a\ generic\ title.=Дополнительные материалы в "Периодическом издании". Этот тип может быть полезен при обращении к таким элементам, как обычные колонки, некрологи, письма к редактору и т.д., которые имеют только общее название. A\ thesis\ written\ for\ an\ educational\ institution\ to\ satisfy\ the\ requirements\ for\ a\ degree.=Тезис, написанный для учебного заведения с целью удовлетворения требований к степени. @@ -2280,10 +2345,6 @@ Removes\ digits.=Удаляет цифры. Presets=Предустановки -Check\ Proxy\ Setting=Проверить настройки прокси -Check\ connection=Проверить подключение -Connection\ failed\!=Соединение потеряно\! -Connection\ successful\!=Соединение успешно\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Создать группы из ключевых слов в следующем поле Generate\ groups\ for\ author\ last\ names=Создание групп для фамилий авторов Regular\ expression=Регулярное выражение @@ -2305,24 +2366,40 @@ Separate\ citations=Разделить цитаты Unprotect\ terms=Незащищенные термины Generate\ a\ new\ key\ for\ imported\ entries\ (overwriting\ their\ default)=Создать новый ключ для импортированных записей (взамен существующих) +Warn\ about\ duplicates\ on\ import=Предупреждать о дубликатах при импорте Import\ and\ Export=Импорт и экспорт Custom\ DOI\ URI=Пользовательский DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=Использовать пользовательский базовый URI от DOI для доступа к материалам Cited\ on\ pages=Цитаты на страницах +Please\ move\ the\ cursor\ into\ the\ document\ text.=Пожалуйста, переместите курсов в текст документа. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=Для получения информации о зрительном положении ваших цитат, мне необходимо передвинуть курсор, однако мне не удалось его захватить. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=Я не могу осуществить вставку в текущее положение курсора. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Пожалуйста, переместите курсор в положение нового цитирования. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Пожалуйста, создайте его в документе или измените в файле\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Пожалуйста, используйте последний вариант в файле оформления ниже во избежание проблем локализации. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' символа %0 является отображаемым именем для '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' символа %0 отсутствует в документе +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' абзаца %0 является отображаемым именем для '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' абзаца %0 отсутствует в документе +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Ошибка при проверке того, записывает ли Writer изменения или содержит записанные изменения. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Невозможно выполнить при включённом [Edit]/[Track Changes]/[Record]. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Изменения, выполненные JabRef, могут привести к неожиданным взаимодействиям с записанными изменениями. +Recording\ and/or\ Recorded\ changes=Изменения записываются и/или записаны +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Сперва используйте [Edit]/[Track Changes]/[Manage], чтобы их разрешить. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=Не удается найти действительный путь сертификации к запрошенной цели (%0), все равно загрузить? Download\ operation\ canceled.=Операция загрузки отменена. @@ -2340,7 +2417,6 @@ Entry\ Type=Тип записи Entry\ types=Типы записей Field\ names=Названия полей Others=Другие -Overwrite\ existing\ field\ values=Перезаписать текущие значения полей Recommended=Рекомендованный Authors\ and\ Title=Авторы и Заголовок @@ -2371,6 +2447,8 @@ Query=Запрос Question=Вопрос Select\ directory=Выберите каталог +Fulltext\ Index=Полнотекстовое индексирование +Automatically\ index\ all\ linked\ files\ for\ fulltext\ search=Автоматически индексировать все связанные файлы для полнотекстового поиска Rebuild\ fulltext\ search\ index=Перестроить полнотекствой индекс Rebuild\ fulltext\ search\ index\ for\ current\ library?=Перестроить полнотекстовой индекс для текущей библиотеки? Rebuilding\ fulltext\ search\ index...=Перестроить полнотекствой индекс... @@ -2383,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Удаленные сервисы Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Разрешить отправку PDF-файлов и цитат в онлайн сервис JabRef (Grobid) для определения метаданных. Это приводит к улучшению результатов. +Fetcher\ cannot\ be\ tested\!=Невозможно проверить сборщик данных\! +Fetcher\ unknown\!=Неизвестный сборщик данных\! Character\ by\ character=Посимвольно Embedded=Встроенный @@ -2397,7 +2477,31 @@ Could\ not\ extract\ Metadata\ from\:\ %0=Не удалось извлечь м Search\ across\ libraries\ in\ a\ new\ window=Поиск по библиотекам в новом окне Keep\ search\ string\ across\ libraries=Сохранить строку поиска по библиотекам Keep\ dialog\ always\ on\ top=Показывать диалог поверх всех окон - - - - +Search\ results\ from\ open\ libraries=Результаты поиска в открытых библиотеках + +Add\ certificate=Добавить сертификат +Serial\ number=Серийный номер +Issuer=Издатель +Valid\ from=Действителен с +Valid\ to=Действителен до +Signature\ algorithm=Алгоритм подписи +Version=Версия + +Error\ downloading=Ошибка загрузки + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Ошибка при записи метаданных. Смотрите журнал ошибок для подробностей. +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Не удалось записать метаданные, файл %1 не найден. +Success\!\ Finished\ writing\ metadata.=Успех\! Запись метаданных завершена. + +Custom\ API\ key=Пользовательский ключ API +Check\ %0\ API\ Key\ Setting=Проверьте настройку ключа API %0 + +Set=Задать +Append=Присоединить +Rename=Переименовать +New\ field\ name=Новое имя поля +Automatic\ field\ editor=Автоматический редактор поля + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Примечание\: Если в исходных записях отсутствуют ключевые слова, подходящие для конфигурации новой группы, то при подтверждении здесь они будут добавлены) +Assign=Назначить +Do\ not\ assign=Не назначать diff --git a/src/main/resources/l10n/JabRef_sv.properties b/src/main/resources/l10n/JabRef_sv.properties index 84779302d46..ae4d6c18d10 100644 --- a/src/main/resources/l10n/JabRef_sv.properties +++ b/src/main/resources/l10n/JabRef_sv.properties @@ -19,6 +19,7 @@ Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=Biblioteke %0/%1\ entries=%0/%1 poster + %0\ matches\ the\ regular\ expression\ %1=%0 matchar det reguljära uttrycket %1 %0\ matches\ the\ term\ %1=%0 matchar termen %1 @@ -138,8 +139,6 @@ Cite\ command=Citeringskommando Clear=Rensa -Clear\ fields=Rensa fält - Open\ /\ close\ entry\ editor=Öppna / stäng postredigerare Close\ dialog=Stäng dialog @@ -618,11 +617,18 @@ Remove\ selected\ entries\ from\ this\ group=Ta bort valda poster från denna gr Remove\ group=Ta bort grupp -Remove\ group\ "%0"?=Ta bort gruppen "%0"? +Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Ta bort gruppen "%0" och dess undergrupper? -Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. + + Remove\ link=Ta bort länk @@ -632,8 +638,6 @@ Remove\ string\ %0=Ta bort sträng %0 Removed\ group\ "%0".=Tog bort gruppen "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. - Removed\ string=Tog bort sträng Renamed\ string=Bytte namn på sträng @@ -685,9 +689,6 @@ Select\ entry\ type=Välj posttyp Select\ file\ from\ ZIP-archive=Välj fil från ZIP-arkiv -Set\ field=Sätt fält -Set\ fields=Sätt fält - Settings=Alternativ Shortcut=Genväg @@ -718,7 +719,6 @@ Size=Storlek Skipped\ -\ No\ PDF\ linked=Hoppade över - Ingen PDF länkad Skipped\ -\ PDF\ does\ not\ exist=Hoppade över - PDF fanns ej -Skipped\ entry.=Hoppade över post. source\ edit=ändring av källkod @@ -828,8 +828,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Rad %0 Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Rad %0\: Hittade korrupt referensnyckel %1 (kommatecken saknas). Download\ from\ URL=Ladda ned från URL Rename\ field=Byt namn på fält -Rename\ field\ to=Byt namn på fält till -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytta innehållet i ett fält till ett fält med ett annat namn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan inte använda port "%0" för fjärråtkomst; ett annat program kanske använder den. Försök med en annan port. @@ -1029,6 +1027,7 @@ Proxy\ requires\ authentication=Proxyn kräver autentisering Attention\:\ Password\ is\ stored\ in\ plain\ text\!=OBS\! Lösenordet sparas i klartext\! Clear\ connection\ settings=Rensa anslutningsinställningar + Open\ folder=Öppna mapp Export\ sort\ order=Soteringsordning vid export @@ -1370,7 +1369,6 @@ Protected\ terms\ file=Fil med skyddade termer Style\ file=Stilfil för OpenOffice/LibreOffice Open\ OpenOffice/LibreOffice\ connection=Öppna OpenOffice/LibreOffice-anslutning -You\ must\ enter\ at\ least\ one\ field\ name=Du måste ange minst ett fältnamn Non-ASCII\ encoded\ character\ found=Bokstäver som inte är ASCII-kodade hittades Toggle\ web\ search\ interface=Växla webbsökning @@ -1571,6 +1569,8 @@ Previous\ preview\ style=Föregående förhandsgranskningsstil + + diff --git a/src/main/resources/l10n/JabRef_tl.properties b/src/main/resources/l10n/JabRef_tl.properties index 0cb29b98e99..de37b3f268a 100644 --- a/src/main/resources/l10n/JabRef_tl.properties +++ b/src/main/resources/l10n/JabRef_tl.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 nag tugma sa regular na ekspresyon %1 %0\ matches\ the\ term\ %1=%0 tumugma sa termino %1 @@ -126,8 +127,6 @@ Cite\ command=Utos na nabanggit Clear=Malinaw -Clear\ fields=Malinaw na patlang - Close\ dialog=Isara ang dialog @@ -600,11 +599,18 @@ Remove\ selected\ entries\ from\ this\ group=Tanggalin ang napiling entries mula Remove\ group=Tanggaling ang grupo -Remove\ group\ "%0"?=Tanggalin ang grupo "%0"? +Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Tanggalin ang grupo "%0" at ang mababang grupo? -Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo +Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. + + Remove\ link=Tanggalin ang link @@ -613,8 +619,6 @@ Remove\ old\ entry=Tanggalin ang lumang entry Removed\ group\ "%0".=Tanggalin ang grupo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. - Removed\ string=Na tanggal ang string Renamed\ string=Palitan ulit ng pangalan ang string @@ -671,9 +675,6 @@ Select\ file\ from\ ZIP-archive=Piliin ang file mula sa na arkiba na ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Piliin ang nodes na puno para makita at matanggap o matanggihan ang pagbabago -Set\ field=Ihanda ang patlang -Set\ fields=Ihanda ang patlang - Settings=Mga settings Shortcut=Shortcut @@ -706,7 +707,6 @@ Size=Laki Skipped\ -\ No\ PDF\ linked=Nilaktawan - Walang PDF ang na linked Skipped\ -\ PDF\ does\ not\ exist=Nilaktawan - PDF ay hindi umiiral -Skipped\ entry.=Nilaktawan ang entry. source\ edit=i-edit ang pinagmulan @@ -820,10 +820,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=I-download mula sa URL Rename\ field=Palitan ang pangalan ng patlang -Append\ field=Ilagay ang field -Append\ to\ fields=Ilagay sa mga patlang -Rename\ field\ to=Palitan ang pangalan ng patlang sa -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Ilipat ang mga nilalaman ng isang patlang sa isang patlang na may ibang pangalan Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Hindi magagamit ang port %0 para sa remote na operasyon; maaaring gamitin ng isa pang application. Subukan ang pagtukoy ng isa pang port. @@ -1033,6 +1029,7 @@ Proxy\ requires\ authentication=Kinakailangan ng proxy ang pagpapatunay Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Pansin\: Ang password ay naka-imbak sa plain text\! Clear\ connection\ settings=I-clear ang mga setting ng koneksyon + Export\ sort\ order=Mag-export ng uri ng order Show\ extra\ columns=Magpakita ng mga dagdag na haligi @@ -1206,7 +1203,6 @@ Protected\ terms\ file=Protektadong mga tuntunin na file Style\ file=Estilo ng file Open\ OpenOffice/LibreOffice\ connection=Buksan ang OpenOffice/LibreOffice na koneksyon -You\ must\ enter\ at\ least\ one\ field\ name=Dapat kang magpasok ng hindi bababa sa isang pangalan ng field Non-ASCII\ encoded\ character\ found=Natagpuan ang naka-encode na karakter na hindi-ASCII Toggle\ web\ search\ interface=I-toggle ang interface ng paghahanap sa web @@ -1276,7 +1272,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Tinanggal ang l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Hindi makuha ang data mula sa entry '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Ang entry mula sa %0 ay hindi ma-parse. Invalid\ identifier\:\ '%0'.=Di-wastong identifier\: '%0'. -This\ paper\ has\ been\ withdrawn.=Ang papel na ito ay na-withdraw. Aux\ file=Aux file Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo na naglalaman ng mga entry na binanggit sa isang ibinigay na file ng TeX @@ -1364,6 +1359,8 @@ Default\ pattern=Default na pattern + + diff --git a/src/main/resources/l10n/JabRef_tr.properties b/src/main/resources/l10n/JabRef_tr.properties index 80d892c18af..fc09ebe0ce1 100644 --- a/src/main/resources/l10n/JabRef_tr.properties +++ b/src/main/resources/l10n/JabRef_tr.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 girdi -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 dışa aktarım başarılı. Kaydedilen dosyayı içeren klasör açılsın mı? + %0\ matches\ the\ regular\ expression\ %1=%0 şu Düzenli İfadeyle eşleşiyor %1 @@ -151,8 +151,6 @@ Cite\ command=Alıntı komutu Clear=Sil -Clear\ fields=Alanları sil - Open\ /\ close\ entry\ editor=Girdi düzenleyicisini aç/kapat Close\ dialog=Dialoğu kapat @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Seçili girdileri bu gruptan çıka Remove\ group=Grubu sil -Remove\ group\ "%0"?="%0" grubu silinsin mi? +Remove\ group\ and\ subgroups=Grubu ve altgrupları sil + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?="%0" grubu ve altgrupları silinsin mi? -Remove\ group\ and\ subgroups=Grubu ve altgrupları sil +Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. + + Remove\ link=Linki sil @@ -717,8 +722,6 @@ Remove\ string\ %0=%0 Dizgesini sil Removed\ group\ "%0".="%0" grubu silindi. -Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. - Removed\ string=Dizge silindi Renamed\ string=Dizge yeniden adlandırıldı @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=ZIP arşivinden dosyayı seçiniz Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Değişiklikleri görmek ve kabul ya da reddetmek için ağaç düğümlerini seçiniz -Set\ field=Alanı ata -Set\ fields=Alanları ata - Settings=Ayarlar Shortcut=Kısayol @@ -825,7 +825,6 @@ Size=Boyut Skipped\ -\ No\ PDF\ linked=Atlandı - PDF eklenmedi Skipped\ -\ PDF\ does\ not\ exist=Atlandı - PDF mevcut değil -Skipped\ entry.=Girdi atlandı. Sort\ subgroups=Alt grupları sırala @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Satır %0\: B No\ full\ text\ document\ found=Tam metin belge bulunamadı Download\ from\ URL=URL'den indir Rename\ field=Alanın adını değiştir -Append\ field=Alan ekle -Append\ to\ fields=Alanlara ekle -Rename\ field\ to=Alan adını şuna değiştir -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Alan içeriğini başka isimli bir alanın içine taşı Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Uzak operasyon için bağlantı noktası %0 kullanılamıyor; bir başka program kullanıyor olabilir. Başka bir bağlantı noktası deneyin. @@ -1243,6 +1238,15 @@ Use\ custom\ proxy\ configuration=Özelleştirilmiş vekil konfigürasyonu kulla Proxy\ requires\ authentication=Vekil sunucu kimlik denetleme gerektiriyor Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Dikkat\: Parola salt metin olarak kaydedildi\! Clear\ connection\ settings=Bağlantı ayarlarını sil +Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et +Check\ connection=Bağlantıyı kontrol et +Connection\ failed\!=Bağlantı başarısız\! +Connection\ successful\!=Bağlantı başarılı\! + +SSL\ Configuration=SSL Yapılandırması +SSL\ certificate\ file=SSL sertifika dosyası +Duplicate\ Certificates=Çift nüsha Sertifikalar +You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Open\ folder=Klasörü aç Export\ sort\ order=Sıralama kriterlerini dışa aktar @@ -1703,7 +1707,6 @@ Protected\ terms\ file=Korunmuş terimler dosyası Style\ file=Stil dosyası Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice bağlantısı aç -You\ must\ enter\ at\ least\ one\ field\ name=En az bir alan adı girmelisiniz Non-ASCII\ encoded\ character\ found=ASCII koduyla kodlanmamış karakter bulundu Non-UTF-8\ encoded\ field\ found=UTF-8 ile kodlanmamış alan bulundu Toggle\ web\ search\ interface=Ağ arama arayüzünü değiştir @@ -1841,7 +1844,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Alan içeriğin Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' dan girdi verileri alınamadı. Entry\ from\ %0\ could\ not\ be\ parsed.=%0'dan girdi çözümlenemedi. Invalid\ identifier\:\ '%0'.=Geçersiz tanımlayıcı\: '%0 '. -This\ paper\ has\ been\ withdrawn.=Bu yayın geri çekildi. empty\ citation\ key=boş atıf anahtarı Aux\ file=Aux dosya Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Belirli bir TeX dosyasında alıntılanmış girdileri içeren grup @@ -2324,10 +2326,6 @@ Removes\ digits.=Rakamları siler. Presets=Ön ayarlar -Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et -Check\ connection=Bağlantıyı kontrol et -Connection\ failed\!=Bağlantı başarısız\! -Connection\ successful\!=Bağlantı başarılı\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Aşağıdaki alandaki anahtar sözcüklerden gruplar oluştur Generate\ groups\ for\ author\ last\ names=Yazar soyadları için gruplar oluştur Regular\ expression=Düzenli ifade @@ -2400,7 +2398,6 @@ Entry\ Type=Girdi Türü Entry\ types=Girdi türleri Field\ names=Alan adları Others=Diğerleri -Overwrite\ existing\ field\ values=Mevcut alan değerlerinin üzerine yaz Recommended=Önerilen Authors\ and\ Title=Yazarlar ve Başlık @@ -2445,6 +2442,8 @@ Grobid\ URL=Grobid URL Remote\ services=Uzak servisler Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Metaverisini belirlemek için PDF dosyalarını ve ham atıf dizgelerini bir JabRef çevrim içi servisine (Grobid) göndermeye izin ver. Bu daha iyi sonuçlara yol açar. +Fetcher\ cannot\ be\ tested\!=Getirici test edilemiyor\! +Fetcher\ unknown\!=Getirici bilinmiyor\! Character\ by\ character=Karakter karakter Embedded=Gömülü @@ -2468,10 +2467,6 @@ Valid\ from=Geçerlilik tarihi Valid\ to=Son kullanma tarihi Signature\ algorithm=İmza algoritması Version=Sürüm -SSL\ Configuration=SSL Yapılandırması -SSL\ certificate\ file=SSL sertifika dosyası -Duplicate\ Certificates=Çift nüsha Sertifikalar -You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Error\ downloading=İndirmede hata @@ -2479,3 +2474,7 @@ Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Metaverisi Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metaverisi yazma başarısız, %1 dosyası bulunamadı. Success\!\ Finished\ writing\ metadata.=Başarılı\! Metaverisi yazma bitirildi. +Custom\ API\ key=Özel API anahtarı +Check\ %0\ API\ Key\ Setting=%0 API Anahtar Ayarlarını Kontrol Edin + + diff --git a/src/main/resources/l10n/JabRef_vi.properties b/src/main/resources/l10n/JabRef_vi.properties index 26aaa90bf04..d82df8e5422 100644 --- a/src/main/resources/l10n/JabRef_vi.properties +++ b/src/main/resources/l10n/JabRef_vi.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 khớp biểu thức chính tắc %1 %0\ matches\ the\ term\ %1=%0 khớp thuật ngữ %1 @@ -133,8 +134,6 @@ Cite\ command=Lệnh trích dẫn Clear=Xóa -Clear\ fields=Xóa các dữ liệu - Close\ dialog=Đóng hộp thoại @@ -612,11 +611,18 @@ Remove\ entry\ from\ import=Loại bỏ mục khỏi lệnh nhập Remove\ group=Loại bỏ nhóm -Remove\ group\ "%0"?=Loại bỏ nhóm "%0"? +Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Loại bỏ nhóm "%0" và các nhóm con của nó? -Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con +Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. + + Remove\ link=Loại bỏ liên kết @@ -625,8 +631,6 @@ Remove\ old\ entry=Loại bỏ mục cũ Removed\ group\ "%0".=Đã loại bỏ nhóm "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. - Removed\ string=Đã loại bỏ chuỗi Renamed\ string=Chuỗi được đặt tên lại @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Chọn tập tin từ tập tin ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Chọn các nốt trên sơ đồ hình cây để xem và chấp nhận hoặc từ chối thay đổi -Set\ field=Thiết lập dữ liệu -Set\ fields=Thiết lập các dữ liệu - Settings=Các thiết lập Shortcut=Phím tắt @@ -712,7 +713,6 @@ Size=Kích thước Skipped\ -\ No\ PDF\ linked=Bị bỏ qua - Không có tập tin PDF được liên kết Skipped\ -\ PDF\ does\ not\ exist=Bỏ qua - tập tin PDF không tồn tại -Skipped\ entry.=Mục bị bỏ qua. source\ edit=chỉnh sửa nguồn @@ -826,8 +826,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Kéo từ URL Rename\ field=Đổi tên dữ liệu -Rename\ field\ to=Đổi tên dữ liệu thành -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Di chuyển nội dung của một dữ liệu sang một dữ liệu có tên khác Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Không thể dùng cổng %0 cho lệnh chạy từ xa; một ứng dụng khác có thể đang dùng nó. Hãy thử chỉ định một cổng khác. @@ -900,6 +898,7 @@ Finished\ automatically\ setting\ external\ links.=Thiết lập tự động c + Open\ folder=Mở thư mục @@ -1135,6 +1134,8 @@ Default\ pattern=Kiểu mặc định + + diff --git a/src/main/resources/l10n/JabRef_zh_CN.properties b/src/main/resources/l10n/JabRef_zh_CN.properties index 0577de6a7b9..baca5d6da46 100644 --- a/src/main/resources/l10n/JabRef_zh_CN.properties +++ b/src/main/resources/l10n/JabRef_zh_CN.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 条目 +Export\ operation\ finished\ successfully.=导出成功。 + +Reveal\ in\ File\ Explorer=在文件夹中显示 %0\ matches\ the\ regular\ expression\ %1=%0 匹配正则表达式 %1 @@ -28,7 +31,7 @@ Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (DEFAULT\ abbreviation)= Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (MEDLINE\ abbreviation)=缩写选中记录的期刊名 (MEDLINE 格式缩写) Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (SHORTEST\ UNIQUE\ abbreviation)=缩写选中记录的期刊名 (最短唯一 (SHORTEST UNIQUE) 格式缩写) -Abbreviate\ names=缩写名 +Abbreviate\ names=缩写名(J. Smith) Abbreviated\ %0\ journal\ names.=缩写的 %0 期刊名称。 Abbreviation=缩写 @@ -150,8 +153,6 @@ Cite\ command=引用命令 Clear=清除 -Clear\ fields=清除字段内容 - Open\ /\ close\ entry\ editor=打开/关闭条目编辑器 Close\ dialog=关闭对话框 @@ -201,7 +202,7 @@ Could\ not\ run\ the\ 'vim'\ program.=无法运行 'vim' 程序。 Could\ not\ save\ file.=无法保存文件 Character\ encoding\ '%0'\ is\ not\ supported.=,不支持编码 '%0'。 -Create\ custom\ fields\ for\ each\ BibTeX\ entry=为每个 BibTeX 条目创建自定义字段 +Create\ custom\ fields\ for\ each\ BibTeX\ entry=Create custom fields for each BibTeX entry crossreferenced\ entries\ included=包含交叉引用的记录 @@ -265,7 +266,7 @@ Display\ help\ on\ command\ line\ options=用命令行选项显示帮助 Display\ only\ entries\ belonging\ to\ all\ selected\ groups=仅显示所选分组的交集条目 Display\ version=显示版本 -Do\ not\ abbreviate\ names=不要缩写姓名 +Do\ not\ abbreviate\ names=不缩写名(John Smith) Do\ not\ import\ entry=不导入记录 @@ -376,9 +377,9 @@ First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=首先选 Fit\ table\ horizontally\ on\ screen=列表宽度填满屏幕宽度 Float=浮动 (结果上浮到最前) -Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式\: 选项卡\: 字段; 字段; ...(例如 General\:url;pdf;note...) +Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式:选项卡\:字段;字段;...(例如:General\:url;pdf;note...) -Format\ of\ author\ and\ editor\ names=作者和编者的姓名格式 +Format\ of\ author\ and\ editor\ names=author和editor的姓名格式 Format\ string=格式化简写字串 Format\ used=使用的格式 @@ -492,7 +493,7 @@ Key\ bindings=热键 Key\ bindings\ changed=热键已更改 -Key\ pattern=键值表达式 +Key\ pattern=Key pattern keys\ in\ library=文献库中的键值 @@ -541,7 +542,9 @@ Moved\ group\ "%0".=移动了分组 "%0"。 Mr.\ DLib\ Privacy\ settings=Mr. DLib 隐私设置 +No\ database\ is\ open=No database is open +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=We need a database to export from. Open one. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=没有收到 Mr. DLib 对于本条目的建议。 @@ -549,9 +552,9 @@ Error\ while\ fetching\ recommendations\ from\ Mr.DLib.=在接收 Mr. DLib 的 Name=名称 -Name\ formatter=姓名格式化器 +Name\ formatter=姓名格式化 -Natbib\ style=Natbib 格式 +Natbib\ style=Natbib样式 nested\ AUX\ files=nested AUX 文件 @@ -700,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=从这个组里移除选定记录 Remove\ group=移除分组 -Remove\ group\ "%0"?=移除分组 "%0"? +Remove\ group\ and\ subgroups=移除分组和子分组 + +Remove\ groups\ and\ subgroups=移除分组和子分组 + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=移除所有选定的分组但保留其子分组? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=移除分组 "%0" 但保留其子分组? + +Remove\ groups=移除分组 + +Removed\ all\ selected\ groups.=移除选中的分组 Remove\ group\ "%0"\ and\ its\ subgroups?=移除分组 "%0" 和它的子分组? -Remove\ group\ and\ subgroups=移除分组和子分组 +Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=移除所有选定的分组及其子分组? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=移除所有选定的分组及其子分组 Remove\ link=移除链接 @@ -714,8 +731,6 @@ Remove\ string\ %0=移除字符串 %0 Removed\ group\ "%0".=已移除分组 "%0"。 -Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 - Removed\ string=已移除简写字串 Renamed\ string=重命名简写字串 @@ -753,7 +768,7 @@ Review=评论 Review\ changes=复查修改 Review\ Field\ Migration=查看字段迁移 -Loading=加载中 +Loading=启动 Save=保存 Save\ all\ finished.=完成保存全部。 @@ -764,10 +779,10 @@ Save\ before\ closing=关闭前保存 Save\ library=保存 Save\ library\ as...=另存为 -Saving=保存中 +Saving=保存 Saving\ all\ libraries...=正在保存所有数据库... Saving\ library=正在保存文献库 -Library\ saved=Library saved +Library\ saved=文献库已保存 Saved\ selected\ to\ '%0'.=保存选中到 '%0'. Search=查找 @@ -787,24 +802,21 @@ Select\ file\ from\ ZIP-archive=从 ZIP-压缩包中选择文件 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=选择树节点查看和接受/拒绝修改 -Set\ field=设置字段内容 -Set\ fields=设置字段内容 - Settings=设置 Shortcut=快捷键 Show/edit\ %0\ source=显示/编辑 %0 源代码 -Show\ 'Firstname\ Lastname'=显示 '名 (Firstname) 姓 (Lastname)' +Show\ 'Firstname\ Lastname'=名在前,姓在后(John Smith) -Show\ 'Lastname,\ Firstname'=显示 '姓 (Lastname), 名 (Firstname)' +Show\ 'Lastname,\ Firstname'=姓在前,名在后(Smith, John) Show\ BibTeX\ source\ by\ default=默认显示 BibTeX 源代码 Show\ confirmation\ dialog\ when\ deleting\ entries=删除多条记录时发出警告 -Show\ last\ names\ only=只显示“姓 (Lastname)” +Show\ last\ names\ only=仅显示姓(Smith) Show\ names\ unchanged=显示原始姓名字串 @@ -822,12 +834,17 @@ Size=大小 Skipped\ -\ No\ PDF\ linked=跳过 - 没有 PDF 链接 Skipped\ -\ PDF\ does\ not\ exist=跳过 - PDF 不存在 -Skipped\ entry.=已跳过记录 +JabRef\ skipped\ the\ entry.=JabRef 跳过了条目。 +Import\ error=导入错误 +Open\ library\ error=打开文献库出错 +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=请检查您的库文件是否存在语法错误。 +SourceTab\ error=SourceTab错误 +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=通过条目编辑器的 "{}bibtex source" 标签中输入时出错。 Sort\ subgroups=排序子组 source\ edit=源代码编辑 -Special\ name\ formatters=特殊的姓名格式化器 +Special\ name\ formatters=姓名格式化 Statically\ group\ entries\ by\ manual\ assignment=手动创建静态分组 @@ -956,7 +973,7 @@ Refuse\ to\ save\ the\ library\ before\ external\ changes\ have\ been\ reviewed. Library\ protection=文献库保护 Unable\ to\ save\ library=无法保存文献库 -Citation\ key\ generator=Citationkey生成器 +Citation\ key\ generator=Citationkey Unable\ to\ open\ link.=无法打开链接。 MIME\ type=MIME 类型 @@ -969,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=第 %0 行\: No\ full\ text\ document\ found=未发现完整的文档 Download\ from\ URL=从 URL 下载 Rename\ field=重命名字段 -Append\ field=添加字段 -Append\ to\ fields=追加到字段 -Rename\ field\ to=重命名字段为 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=将某个字段的内容移动到另一个字段中 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=无法使用端口 %0 进行远程操作;该端口可能被其它应用程序占用,请使用其它端口。 @@ -1042,6 +1055,7 @@ Cite\ special=引用特殊 Extra\ information\ (e.g.\ page\ number)=额外信息(例如:页码) Manage\ citations=管理文献引用 Problem\ modifying\ citation=修改文献引用存在问题 +Problem\ collecting\ citations=Problem collecting citations Citation=文献引用 Connecting...=连接中... Select\ style=选择引用样式 @@ -1056,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=同步 OpenOffice/LibreOffice 参考 Select\ which\ open\ Writer\ document\ to\ work\ on=选择使用哪个打开的 Writer 文档 Connected\ to\ document=连接到文档 +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Could not connect to any Writer document. Please make sure a document is open before using the 'Select Writer document' button to connect to it. +No\ Writer\ documents\ found=No Writer documents found Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=插入一条没有内容的引用(这条记录将会出现在引用列表中) Cite\ selected\ entries\ with\ extra\ information=引用包含额外信息的选中记录 @@ -1064,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=保证参考文献是最新的 Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=您的OpenOffice/LibreOffice文档引用了 citationkey "%0",但未在当前库中找到。 +This\ operation\ requires\ a\ bibliography\ database.=This operation requires a bibliography database. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Your OpenOffice/LibreOffice document references at least %0 citation keys which could not be found in your current library. Some of these are %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=Your OpenOffice/LibreOffice document references no citation keys which could also be found in your current library. Unable\ to\ synchronize\ bibliography=无法同步参考文献 Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=合并仅仅由空格分隔的两段引文 @@ -1073,7 +1092,12 @@ Autodetection\ failed=自动检测失败 Please\ wait...=请稍候... Connection\ lost=连接丢失 +Could\ not\ update\ bibliography=Could not update bibliography +Not\ connected\ to\ document=Not connected to document +Problem\ combining\ cite\ markers=Problem combining cite markers +Unable\ to\ reload\ style\ file=Unable to reload style file +Problem\ during\ separating\ cite\ markers=Problem during separating cite markers Automatically\ sync\ bibliography\ when\ inserting\ citations=当插入文献引用时自动同步参考文献 Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=在当前标签查找 BibTeX 记录 @@ -1176,6 +1200,7 @@ Toggle\ relevance=标记为相关 Toggle\ quality\ assured=标记为质量已确认 Toggle\ print\ status=标记打印状态 Update\ keywords=更新关键词 +Problem\ connecting=Problem connecting Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=与OpenOffice/LibreOffice的连接已丢失。请确保OpenOffice/LibreOffice正在运行,并尝试重新连接。 JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef 将会为每个条目发送至少一个请求至发布者。 @@ -1228,6 +1253,16 @@ Use\ custom\ proxy\ configuration=使用自定义 HTTP 代理 Proxy\ requires\ authentication=代理服务器需要密码 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意\: 密码以明文形式保存\! Clear\ connection\ settings=清除连接设置 +Check\ Proxy\ Setting=检查代理设置 +Check\ connection=检查连接 +Connection\ failed\!=连接失败! +Connection\ successful\!=连接成功! + +SSL\ Configuration=SSL配置 +SSL\ configuration\ changed=SSL配置已更改 +SSL\ certificate\ file=SSL证书文件 +Duplicate\ Certificates=复制证书 +You\ already\ added\ this\ certificate=您已经添加了此证书 Open\ folder=打开文件夹 Export\ sort\ order=导出顺序 @@ -1264,6 +1299,7 @@ Please\ open\ %0\ manually.=请手动打开 %0。 The\ link\ has\ been\ copied\ to\ the\ clipboard.=链接已复制到剪贴板 Open\ %0\ file=打开文件 %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=无法自动检测终端。请在首选项中自定义一个终端。 Cannot\ delete\ file=无法删除文件 File\ permission\ error=文件权限错误 @@ -1365,7 +1401,7 @@ Automatically\ setting\ file\ links=自动设置文件链接 Regenerating\ citation\ keys\ according\ to\ metadata=根据元数据重新生成 citationkey Regenerate\ all\ keys\ for\ the\ entries\ in\ a\ BibTeX\ file=重新生成 BibTeX 文件中所有记录的键值 Show\ debug\ level\ messages=显示调试级别消息 -Default\ library\ mode=Default library mode +Default\ library\ mode=默认文献库模式 Show\ only\ preferences\ deviating\ from\ their\ default\ value=只显示与默认值不同的首选项 default=默认 key=键值 @@ -1570,10 +1606,21 @@ Custom=自定义 Export\ cited=导出已被引用的 Unable\ to\ generate\ new\ library=无法生成新库 +The\ cursor\ is\ in\ a\ protected\ area.=The cursor is in a protected area. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=No bibliography database is open for citation. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=No database is open for updating citation markers after citing. +No\ bibliography\ entries\ are\ selected\ for\ citation.=No bibliography entries are selected for citation. +No\ bibliography\ style\ is\ selected\ for\ citation.=No bibliography style is selected for citation. +No\ database=No database +No\ entries\ selected\ for\ citation=No entries selected for citation +Open\ one\ before\ citing.=Open one before citing. +Select\ one\ before\ citing.=Select one before citing. +Select\ some\ before\ citing.=Select some before citing. Found\ identical\ ranges=Found identical ranges Found\ overlapping\ ranges=Found overlapping ranges @@ -1621,7 +1668,7 @@ Remove\ list=删除列表 Add\ abbreviation=添加缩写 Full\ journal\ name=期刊全称 Abbreviation\ name=缩写名 -Shortest\ unique\ abbreviation=Shortest unique abbreviation +Shortest\ unique\ abbreviation=最简缩写 No\ abbreviation\ files\ loaded=缩写文件未加载 @@ -1640,6 +1687,8 @@ Issue\ report\ successful=问题报告成功 Your\ issue\ was\ reported\ in\ your\ browser.=您的问题已在浏览器中报告。 The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=日志和异常信息已复制到剪贴板中。 Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=请将此信息粘贴至问题描述中 (用 Ctrl + V)。 +Last\ notification=最新通知 +Check\ the\ event\ log\ to\ see\ all\ notifications=在事件日志中查看所有通知 Host=主机 Port=端口 @@ -1677,7 +1726,6 @@ Protected\ terms\ file=受保护的术语文件 Style\ file=样式文件 Open\ OpenOffice/LibreOffice\ connection=打开 OpenOffice/LibreOffice 连接 -You\ must\ enter\ at\ least\ one\ field\ name=您至少需要输入一个字段名 Non-ASCII\ encoded\ character\ found=发现Non-ASCII编码字符 Non-UTF-8\ encoded\ field\ found=未找到 UTF-8 编码的字段 Toggle\ web\ search\ interface=切换网页搜索面板 @@ -1758,7 +1806,7 @@ Unhandled\ exception\ occurred.=发生了无法处理的异常。 strings\ included=包含的字符串 Escape\ underscores=转义下划线 -Color=色彩 +Color=颜色 Please\ also\ add\ all\ steps\ to\ reproduce\ this\ issue,\ if\ possible.=如果可能的话,也请添加所有步骤以重现这个问题。 Fit\ width=适应宽度 Fit\ a\ single\ page=适应单页 @@ -1815,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=移除字段内 Could\ not\ retrieve\ entry\ data\ from\ '%0'.=无法从 "%0" 中检索条目数据。 Entry\ from\ %0\ could\ not\ be\ parsed.=无法解析 %0 中的条目。 Invalid\ identifier\:\ '%0'.=无效的标识符:'%0'。 -This\ paper\ has\ been\ withdrawn.=这篇论文已被撤回。 empty\ citation\ key=Citationkey为空 Aux\ file=Aux 文件 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=包含给定TeX文件中引用的条目组 @@ -1883,7 +1930,7 @@ View\ change\ log=查看变更记录 View\ event\ log=查看事件日志 Website=网站 -Override\ default\ font\ settings=跳过默认字体设置 +Override\ default\ font\ settings=覆盖默认的字体设置 Clear\ search=清除搜索 Click\ help\ to\ learn\ about\ the\ migration\ of\ pre-3.6\ databases.=点击帮助了解 pre-3.6 数据库的迁移情况。 @@ -1993,47 +2040,47 @@ Import\ entries\ from\ LaTeX\ files=Import entries from LaTeX files Import\ new\ entries=Import new entries Group\ color=分组颜色 -Columns=Columns +Columns=列 File\ type=File type IEEE=IEEE Internal=Internal Special=Special Remove\ column=Remove column -Add\ custom\ column=Add custom column -Update\ to\ current\ column\ order=Update to current column order -Sort\ column\ one\ step\ upwards=Sort column one step upwards -Sort\ column\ one\ step\ downwards=Sort column one step downwards +Add\ custom\ column=添加自定义列 +Update\ to\ current\ column\ order=更新到当前的列顺序 +Sort\ column\ one\ step\ upwards=上移一位 +Sort\ column\ one\ step\ downwards=下移一位 List\ must\ not\ be\ empty.=List must not be empty. -Order=Order +Order=排序 -Add\ field\ to\ filter\ list=Add field to filter list +Add\ field\ to\ filter\ list=添加字段到过滤列表 Add\ formatter\ to\ list=Add formatter to list Filter\ List=Filter List Open\ files...=打开文件... Affected\ fields\:=补全字段 -Show\ preview\ as\ a\ tab\ in\ entry\ editor=Show preview as a tab in entry editor -Font=Font -Visual\ theme=Visual theme +Show\ preview\ as\ a\ tab\ in\ entry\ editor=在条目编辑器中以标签形式显示预览 +Font=字体 +Visual\ theme=主题风格 Light\ theme=明亮主题 Dark\ theme=暗色主题 Custom\ theme=自定义 Overwrite\ existing\ keys=覆盖该字段的内容 -Key\ patterns=不同类型条目的键值风格 +Key\ patterns=Citationkey样式 Font\ settings=Font settings Please\ specify\ a\ css\ theme\ file.=请指定一个 css 主题文件。 You\ must\ enter\ an\ integer\ value\ higher\ than\ 8.=You must enter an integer value higher than 8. Letters\ after\ duplicate\ generated\ keys=为重复的Citationkey添加字母后缀 -Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 A (a, b, ...) -Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 B (b, c, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 a (a, b, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 b (b, c, ...) Always\ add\ letter\ (a,\ b,\ ...)\ to\ generated\ keys=在生成键值时总是添加字母 (a, b, ...) -Default\ pattern=默认模式 +Default\ pattern=Default pattern Reset\ %s\ to\ default\ value=Reset %s to default value Library\ mode=Library mode Reset\ to\ recommended=Reset to recommended Remove\ all=Remove all -Reset\ All=Reset All +Reset\ All=全部重置 Linked\ identifiers=Linked identifiers insert\ entries=insert entries @@ -2056,13 +2103,14 @@ Explicit\ selection=手动选择 Searching\ for\ a\ keyword=指定字段的关键词 Free\ search\ expression=检索语句 Specified\ keywords=字段内容 -Cited\ entries=Cited entries +Cited\ entries=根据AUX文件 Search\ term\ is\ empty.=Search term is empty. Invalid\ regular\ expression.=无效的正则表达式 Please\ provide\ a\ valid\ aux\ file.=请提供一个有效的AUX文件 Keyword\ delimiter=Keyword delimiter Hierarchical\ keyword\ delimiter=Hierarchical keyword delimiter Escape\ ampersands=转义逻辑"与"符号 +Escape\ dollar\ sign=Escape dollar sign Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Hint\:\n\nTo search all fields for Smith, enter\:\nsmith\n\nTo search the field author for Smith and the field title for electrical, enter\:\nauthor\=Smith and title\=electrical @@ -2272,7 +2320,7 @@ Reveal\ in\ file\ explorer=Reveal in file explorer Autolink\ files=自动链接文件 -Custom\ editor\ tabs=条目编辑器的标签 +Custom\ editor\ tabs=条目编辑器 Custom\ export\ formats=自定义导出格式 Custom\ import\ formats=自定义导入格式 @@ -2283,7 +2331,7 @@ Customized\ preview\ style=自定义预览样式 Next\ preview\ style=下一个预览样式 Previous\ preview\ style=上一个预览样式 -(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=( Note\: Press return to commit changes in the table\! ) +(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=(注意:按回车键以确认更改) Reset=Reset New\ inproceedings=新的inproceedings Reset\ entry\ types\ and\ fields\ to\ defaults=Reset entry types and fields to defaults @@ -2297,10 +2345,6 @@ Removes\ digits.=Removes digits. Presets=默认 -Check\ Proxy\ Setting=检查代理设置 -Check\ connection=检查连接 -Connection\ failed\!=连接失败! -Connection\ successful\!=连接成功! Generate\ groups\ from\ keywords\ in\ the\ following\ field=根据以下字段的关键字生成组 Generate\ groups\ for\ author\ last\ names=根据作者的姓氏生成组 Regular\ expression=正则表达式 @@ -2328,19 +2372,34 @@ Custom\ DOI\ URI=自定义 DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=使用基于DOI的URI访问文章 Cited\ on\ pages=Cited on pages +Please\ move\ the\ cursor\ into\ the\ document\ text.=Please move the cursor into the document text. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=To get the visual positions of your citations I need to move the cursor around, but could not get it. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=I cannot insert to the cursors current location. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Please move the cursor to the location for the new citation. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Please create it in the document or change in the file\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Please use the latter in the style file below to avoid localization problems. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 character style '%1' is a display name for '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 character style '%1' is missing from the document +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 paragraph style '%1' is a display name for '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 paragraph style '%1' is missing from the document +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Error while checking if Writer is recording changes or has recorded changes. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Cannot work with [Edit]/[Track Changes]/[Record] turned on. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Changes by JabRef could result in unexpected interactions with recorded changes. +Recording\ and/or\ Recorded\ changes=Recording and/or Recorded changes +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Use [Edit]/[Track Changes]/[Manage] to resolve them first. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=无法找到请求目标(%0)的有效证书路径,仍然下载? Download\ operation\ canceled.=下载操作已取消。 @@ -2358,7 +2417,6 @@ Entry\ Type=条目类型 Entry\ types=条目类型 Field\ names=字段名 Others=其他 -Overwrite\ existing\ field\ values=覆盖现有字段内容 Recommended=推荐 Authors\ and\ Title=作者和标题 @@ -2401,8 +2459,10 @@ Found\ matches\ in\ Annotations\:=在注释中找到匹配项: Grobid\ URL=Grobid URL Remote\ services=远程服务 -Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Allow sending PDF files and raw citation strings to a JabRef online service (Grobid) to determine Metadata. This produces better results. +Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=允许将 PDF 文件和原始引用发送到JabRef在线服务(Grobid)以获取元数据 +Fetcher\ cannot\ be\ tested\!=Fetcher cannot be tested\! +Fetcher\ unknown\!=Fetcher unknown\! Character\ by\ character=Character by character Embedded=内嵌 @@ -2419,6 +2479,29 @@ Keep\ search\ string\ across\ libraries=在所有库中保留搜索语句 Keep\ dialog\ always\ on\ top=保持窗口位于最前面 Search\ results\ from\ open\ libraries=Search results from open libraries +Add\ certificate=添加证书 +Serial\ number=序号 +Issuer=签发者 +Valid\ from=起始有效期 +Valid\ to=截至有效期 +Signature\ algorithm=签名算法 +Version=版本 + +Error\ downloading=下载出错 + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=写入元数据时出错。详细信息请参阅错误日志。 +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Failed to write metadata, file %1 not found. +Success\!\ Finished\ writing\ metadata.=Success\! Finished writing metadata. +Custom\ API\ key=自定义API +Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting +Set=设定 +Append=附加 +Rename=重命名 +New\ field\ name=新的字段名 +Automatic\ field\ editor=自动化条目编辑 +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note\: If original entries lack keywords to qualify for the new group configuration, confirming here will add them) +Assign=Assign +Do\ not\ assign=Do not assign diff --git a/src/main/resources/l10n/JabRef_zh_TW.properties b/src/main/resources/l10n/JabRef_zh_TW.properties index 588f8dd4d77..288532048c0 100644 --- a/src/main/resources/l10n/JabRef_zh_TW.properties +++ b/src/main/resources/l10n/JabRef_zh_TW.properties @@ -15,6 +15,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 條目 + %0\ matches\ the\ regular\ expression\ %1=%0 符合正規表達式 %1 %0\ matches\ the\ term\ %1=%0 符合字詞 %1 @@ -142,8 +143,6 @@ Cite\ command=引用指令 Clear=清除 -Clear\ fields=清除欄位 - Close\ dialog=關閉對話視窗 @@ -579,6 +578,13 @@ Remove\ group=移除群組 + + + + + + + Remove\ link=移除連結 Remove\ old\ entry=移除舊有條目 @@ -588,7 +594,6 @@ Remove\ string\ %0=移除字串 %0 - Replace=取代 Replace\ With\:=取代為: Find\:=尋找: @@ -627,7 +632,6 @@ Select\ all=全選 - Settings=設定 Shortcut=快速鍵 @@ -728,8 +732,6 @@ Unable\ to\ save\ library=無法儲存文獻倉庫 Rename\ field=重新命名欄位 -Append\ field=添加欄位 -Append\ to\ fields=添加到欄位 Autosave\ local\ libraries=自動儲存本機文獻倉庫 @@ -806,6 +808,11 @@ Proxy\ configuration=代理伺服器設定 Use\ custom\ proxy\ configuration=使用自訂代理伺服器 Proxy\ requires\ authentication=代理伺服器需要認證 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意:密碼將以明文形式存放! +Check\ Proxy\ Setting=檢查代理設定 +Check\ connection=檢查連接 +Connection\ failed\!=連線失敗! +Connection\ successful\!=連線成功! + Export\ sort\ order=匯出順序 @@ -984,7 +991,7 @@ Set\ rank\ to\ five=設定評分為 5 級 Order=順序 -Affected\ fields\:=影響欄位\: +Affected\ fields\:=影響欄位\: Show\ preview\ as\ a\ tab\ in\ entry\ editor=在條目編輯器中以分頁形式顯示預覽 Font=字型 Visual\ theme=界面主題 @@ -1056,10 +1063,6 @@ Remove\ digits=移除數字 Removes\ digits.=移除數字 -Check\ Proxy\ Setting=檢查代理設定 -Check\ connection=檢查連接 -Connection\ failed\!=連線失敗! -Connection\ successful\!=連線成功! Generate\ groups\ for\ author\ last\ names=依據作者姓氏生成群組 Regular\ expression=正規表達式 @@ -1102,3 +1105,5 @@ Select\ directory=選擇資料夾 + + diff --git a/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..75a1a176597 --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java @@ -0,0 +1,105 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.copyormovecontent.CopyOrMoveFieldContentTabViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class CopyOrMoveFieldContentTabViewModelTest { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998"); + bibDatabase = new BibDatabase(); + copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + } + + @Test + void copyValueDoesNotCopyBlankValues() { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + copyOrMoveFieldContentTabViewModel.copyValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.DATE), "YEAR field is not copied correctly to the DATE field"); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR), "YEAR field should not have changed"); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE), "DATE field should not have changed because the YEAR field is blank e.g it doesn't exist"); + } + + @Test + void swapValuesShouldNotSwapFieldValuesIfOneOfTheValuesIsBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE)); + assertEquals(Optional.empty(), entryB.getField(StandardField.YEAR)); + } + + @Test + void swapValuesShouldSwapFieldValuesIfBothValuesAreNotBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(List.of(Optional.of("2014"), Optional.of("2015")), + List.of(entryA.getField(StandardField.YEAR), entryA.getField(StandardField.DATE)), + "YEAR and DATE values didn't swap"); + } + + @Test + void moveValueShouldNotMoveValueIfToFieldIsNotBlankAndOverwriteIsNotEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(false); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + } + + @Test + void moveValueShouldMoveValueIfOverwriteIsEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + private CopyOrMoveFieldContentTabViewModel newTwoFieldsViewModel(BibEntry... selectedEntries) { + return new CopyOrMoveFieldContentTabViewModel(List.of(selectedEntries), bibDatabase, stateManager); + } +} diff --git a/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..1221580762a --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java @@ -0,0 +1,118 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.editfieldcontent.EditFieldContentViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotEquals; +import static org.mockito.Mockito.mock; + +public class EditFieldContentTabViewModelTest { + EditFieldContentViewModel editFieldContentViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, ""); + + bibDatabase = new BibDatabase(); + editFieldContentViewModel = new EditFieldContentViewModel(bibDatabase, List.of(entryA, entryB), stateManager); + } + + @Test + void clearSelectedFieldShouldClearFieldContentEvenWhenOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.YEAR)); + } + + @Test + void clearSelectedFieldShouldDoNothingWhenFieldDoesntExistOrIsEmpty() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.FILE); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.FILE)); + } + + @Test + void setFieldValueShouldNotDoAnythingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfFieldContentIsEmpty() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryB.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldDoNothingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldAppendFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("20150"), entryA.getField(StandardField.YEAR)); + } + + @Test + void getAllFieldsShouldNeverBeEmpty() { + assertNotEquals(0, editFieldContentViewModel.getAllFields().size()); + } + + @Test + void getSelectedFieldShouldHaveADefaultValue() { + assertNotEquals(null, editFieldContentViewModel.getSelectedField()); + } +} diff --git a/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java new file mode 100644 index 00000000000..bf5a246e97f --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java @@ -0,0 +1,111 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.renamefield.RenameFieldViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.field.UnknownField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class RenameFieldViewModelTest { + RenameFieldViewModel renameFieldViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014") + .withField(StandardField.AUTHOR, "Doe"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, "") + .withField(StandardField.AUTHOR, "Eddie"); + + bibDatabase = new BibDatabase(); + renameFieldViewModel = new RenameFieldViewModel(List.of(entryA, entryB), bibDatabase, stateManager); + } + + @Test + void renameFieldShouldRenameFieldIfItExist() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName("ETAD"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryA.getField(StandardField.DATE)); + + assertEquals(Optional.of("1998"), entryB.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + @Test + void renameFieldShouldDoNothingIfFieldDoNotExist() { + Field toRenameField = new UnknownField("Some_field_that_doesnt_exist"); + renameFieldViewModel.selectField(toRenameField); + renameFieldViewModel.setNewFieldName("new_field_name"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.empty(), entryA.getField(toRenameField)); + assertEquals(Optional.empty(), entryA.getField(new UnknownField("new_field_name"))); + + assertEquals(Optional.empty(), entryB.getField(toRenameField)); + assertEquals(Optional.empty(), entryB.getField(new UnknownField("new_field_name"))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameIsEmpty() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName(""); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField(""))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField(""))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameHasWhitespaceCharacters() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName("Hello, World"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField("Hello, World"))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField("Hello, World"))); + } + + @Test + void renameFieldShouldDoNothingWhenThereIsAlreadyAFieldWithTheSameNameAsNewFieldName() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName(StandardField.YEAR.getName()); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + } +} diff --git a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java index 1f1c20fc329..9fded7c7802 100644 --- a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java +++ b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java @@ -56,7 +56,7 @@ void compareWithChangedGroup() { Optional groupDiff = GroupDiff.compare(originalMetaData, newMetaData); - Optional expectedGroupDiff = Optional.of(new GroupDiff(newMetaData.getGroups().get(), originalMetaData.getGroups().get())); + Optional expectedGroupDiff = Optional.of(new GroupDiff(originalMetaData.getGroups().get(), newMetaData.getGroups().get())); assertEquals(expectedGroupDiff.get().getNewGroupRoot(), groupDiff.get().getNewGroupRoot()); assertEquals(expectedGroupDiff.get().getOriginalGroupRoot(), groupDiff.get().getOriginalGroupRoot()); diff --git a/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java new file mode 100644 index 00000000000..1b63e26cbb3 --- /dev/null +++ b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java @@ -0,0 +1,666 @@ +package org.jabref.logic.bst; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.jabref.logic.bst.util.BstCaseChangersTest; +import org.jabref.logic.bst.util.BstNameFormatterTest; +import org.jabref.logic.bst.util.BstPurifierTest; +import org.jabref.logic.bst.util.BstTextPrefixerTest; +import org.jabref.logic.bst.util.BstWidthCalculatorTest; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.types.StandardEntryType; + +import org.antlr.v4.runtime.RecognitionException; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +/** + * For additional tests see for + * + * purify: {@link BstPurifierTest} + * width: {@link BstWidthCalculatorTest} + * format.name: {@link BstNameFormatterTest} + * change.case: {@link BstCaseChangersTest} + * prefix: {@link BstTextPrefixerTest} + * + */ +class BstFunctionsTest { + @Test + public void testCompareFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test.compare } { + #5 #5 = % TRUE + #1 #2 = % FALSE + #3 #4 < % TRUE + #4 #3 < % FALSE + #4 #4 < % FALSE + #3 #4 > % FALSE + #4 #3 > % TRUE + #4 #4 > % FALSE + "H" "H" = % TRUE + "H" "Ha" = % FALSE + } + EXECUTE { test.compare } + """); + + vm.render(Collections.emptyList()); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 #1 + % 2 + #5 #2 - % 3 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(3, vm.getStack().pop()); + assertEquals(2, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctionTypeMismatch() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 "HELLO" + % Should throw exception + } + EXECUTE { test } + """); + + assertThrows(BstVMException.class, () -> vm.render(Collections.emptyList())); + } + + @Test + public void testStringOperations() throws RecognitionException { + // Test for concat (*) and add.period + BstVM vm = new BstVM(""" + FUNCTION { test } { + "H" "ello" * % Hello + "Johnny" add.period$ % Johnny. + "Johnny." add.period$ % Johnny. + "Johnny!" add.period$ % Johnny! + "Johnny?" add.period$ % Johnny? + "Johnny} }}}" add.period$ % Johnny.} + "Johnny!}" add.period$ % Johnny!} + "Johnny?}" add.period$ % Johnny?} + "Johnny.}" add.period$ % Johnny.} + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?}", vm.getStack().pop()); + assertEquals("Johnny!}", vm.getStack().pop()); + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?", vm.getStack().pop()); + assertEquals("Johnny!", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Hello", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testMissing() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { test } { title missing$ cite$ } + ITERATE { test } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Article) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "No title")); + + vm.render(testEntries); + + assertEquals("test", vm.getStack().pop()); // cite + assertEquals(BstVM.TRUE, vm.getStack().pop()); // missing title + assertEquals("canh05", vm.getStack().pop()); // cite + assertEquals(BstVM.FALSE, vm.getStack().pop()); // missing title + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testNumNames() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "Johnny Foo { and } Mary Bar" num.names$ + "Johnny Foo and Mary Bar" num.names$ + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(2, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSubstring() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "123456789" #2 #1 substring$ % 2 + "123456789" #4 global.max$ substring$ % 456789 + "123456789" #1 #9 substring$ % 123456789 + "123456789" #1 #10 substring$ % 123456789 + "123456789" #1 #99 substring$ % 123456789 + "123456789" #-7 #3 substring$ % 123 + "123456789" #-1 #1 substring$ % 9 + "123456789" #-1 #3 substring$ % 789 + "123456789" #-2 #2 substring$ % 78 + } EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("78", vm.getStack().pop()); + assertEquals("789", vm.getStack().pop()); + assertEquals("9", vm.getStack().pop()); + assertEquals("123", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("456789", vm.getStack().pop()); + assertEquals("2", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testEmpty() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + READ + STRINGS { s } + FUNCTION { test } { + s empty$ % TRUE + "" empty$ % TRUE + " " empty$ % TRUE + title empty$ % TRUE + " HALLO " empty$ % FALSE + } + ITERATE { test } + """); + List testEntry = List.of(new BibEntry(StandardEntryType.Article)); + + vm.render(testEntry); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameStatic() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { format }{ "Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin" #1 "{vv~}{ll}{, jj}{, f}?" format.name$ } + EXECUTE { format } + """); + List v = Collections.emptyList(); + + vm.render(v); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameInEntries() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { author } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { format }{ author #2 "{vv~}{ll}{, jj}{, f}?" format.name$ } + ITERATE { format } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "Jonathan Meyer and Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin")); + + vm.render(testEntries); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals("Annabi, H?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChangeCase() throws RecognitionException { + BstVM vm = new BstVM(""" + STRINGS { title } + READ + FUNCTION { format.title } { + duplicate$ empty$ + { pop$ "" } + { "t" change.case$ } + if$ + } + FUNCTION { test } { + "hello world" "u" change.case$ format.title + "Hello World" format.title + "" format.title + "{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase" "u" change.case$ format.title + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase", + vm.getStack().pop()); + assertEquals("", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testTextLength() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "hello world" text.length$ % 11 + "Hello {W}orld" text.length$ % 11 + "" text.length$ % 0 + "{A}{D}/{Cycle}" text.length$ % 8 + "{\\This is one character}" text.length$ % 1 + "{\\This {is} {one} {c{h}}aracter as well}" text.length$ % 1 + "{\\And this too" text.length$ % 1 + "These are {\\11}" text.length$ % 11 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(11, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(8, vm.getStack().pop()); + assertEquals(0, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testIntToStr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { #3 int.to.str$ #9999 int.to.str$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("9999", vm.getStack().pop()); + assertEquals("3", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToInt() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(72, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToIntIntToChr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ int.to.chr$ } + EXECUTE {test} + """); + + vm.render(Collections.emptyList()); + + assertEquals("H", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + SORT + FUNCTION { test } { type$ } + ITERATE { test } + """); + List testEntries = List.of( + new BibEntry(StandardEntryType.Article).withCitationKey("a"), + new BibEntry(StandardEntryType.Book).withCitationKey("b"), + new BibEntry(StandardEntryType.Misc).withCitationKey("c"), + new BibEntry(StandardEntryType.InProceedings).withCitationKey("d")); + + vm.render(testEntries); + + assertEquals("inproceedings", vm.getStack().pop()); + assertEquals("misc", vm.getStack().pop()); + assertEquals("book", vm.getStack().pop()); + assertEquals("article", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testCallType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { inproceedings }{ "InProceedings called on " title * } + FUNCTION { book }{ "Book called on " title * } + ITERATE { call.type$ } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.TITLE, "Test")); + + vm.render(testEntries); + + assertEquals("Book called on Test", vm.getStack().pop()); + assertEquals( + "InProceedings called on Effective work practices for floss development: A model and propositions", + vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSwap() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { a } { #3 "Hallo" swap$ } + EXECUTE { a } + """); + + List
- * Documentation can be found in the original bibtex distribution: - *
- * https://www.ctan.org/pkg/bibtex - */ -public class VM implements Warn { - - public static final Integer FALSE = 0; - - public static final Integer TRUE = 1; - - private static final Pattern ADD_PERIOD_PATTERN = Pattern.compile("([^\\.\\?\\!\\}\\s])(\\}|\\s)*$"); - - private static final Logger LOGGER = LoggerFactory.getLogger(VM.class); - - private List entries; - - private Map strings = new HashMap<>(); - - private Map integers = new HashMap<>(); - - private Map functions = new HashMap<>(); - - private Stack stack = new Stack<>(); - - private final Map buildInFunctions; - - private File file; - - private final CommonTree tree; - - private StringBuilder bbl; - - private String preamble = ""; - - public static class Identifier { - - public final String name; - - public Identifier(String name) { - this.name = name; - } - - public String getName() { - return name; - } - } - - public static class Variable { - - public final String name; - - public Variable(String name) { - this.name = name; - } - - public String getName() { - return name; - } - } - - @FunctionalInterface - public interface BstFunction { - void execute(BstEntry context); - } - - public VM(File f) throws RecognitionException, IOException { - this(new ANTLRFileStream(f.getPath())); - this.file = f; - } - - public VM(String s) throws RecognitionException { - this(new ANTLRStringStream(s)); - } - - private VM(CharStream bst) throws RecognitionException { - this(VM.charStream2CommonTree(bst)); - } - - private VM(CommonTree tree) { - this.tree = tree; - - this.buildInFunctions = new HashMap<>(37); - - /* - * Pops the top two (integer) literals, compares them, and pushes - * the integer 1 if the second is greater than the first, 0 - * otherwise. - */ - buildInFunctions.put(">", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation >"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only compare two integers with >"); - } - - stack.push(((Integer) o1).compareTo((Integer) o2) > 0 ? VM.TRUE : VM.FALSE); - }); - - /* Analogous to >. */ - buildInFunctions.put("<", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation <"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only compare two integers with <"); - } - - stack.push(((Integer) o1).compareTo((Integer) o2) < 0 ? VM.TRUE : VM.FALSE); - }); - - /* - * Pops the top two (both integer or both string) literals, compares - * them, and pushes the integer 1 if they're equal, 0 otherwise. - */ - buildInFunctions.put("=", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation ="); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - - if ((o1 == null) ^ (o2 == null)) { - stack.push(VM.FALSE); - return; - } - - if ((o1 == null) && (o2 == null)) { - stack.push(VM.TRUE); - return; - } - - stack.push(o1.equals(o2) ? VM.TRUE : VM.FALSE); - }); - - /* Pops the top two (integer) literals and pushes their sum. */ - buildInFunctions.put("+", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation +"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only compare two integers with +"); - } - - stack.push((Integer) o1 + (Integer) o2); - }); - - /* - * Pops the top two (integer) literals and pushes their difference - * (the first subtracted from the second). - */ - buildInFunctions.put("-", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation -"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer))) { - throw new VMException("Can only subtract two integers with -"); - } - - stack.push((Integer) o1 - (Integer) o2); - }); - - /* - * Pops the top two (string) literals, concatenates them (in reverse - * order, that is, the order in which pushed), and pushes the - * resulting string. - */ - buildInFunctions.put("*", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation *"); - } - Object o2 = stack.pop(); - Object o1 = stack.pop(); - - if (o1 == null) { - o1 = ""; - } - if (o2 == null) { - o2 = ""; - } - - if (!((o1 instanceof String) && (o2 instanceof String))) { - LOGGER.error("o1: {} ({})", o1, o1.getClass()); - LOGGER.error("o2: {} ({})", o2, o2.getClass()); - throw new VMException("Can only concatenate two String with *"); - } - - stack.push(o1.toString() + o2); - }); - - /* - * Pops the top two literals and assigns to the first (which must be - * a global or entry variable) the value of the second. - */ - buildInFunctions.put(":=", context -> { - if (stack.size() < 2) { - throw new VMException("Invalid call to operation :="); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - assign(context, o1, o2); - }); - - /* - * Pops the top (string) literal, adds a `.' to it if the last non - * '}' character isn't a `.', `?', or `!', and pushes this resulting - * string. - */ - buildInFunctions.put("add.period$", context -> addPeriodFunction()); - - /* - * Executes the function whose name is the entry type of an entry. - * For example if an entry is of type book, this function executes - * the book function. When given as an argument to the ITERATE - * command, call.type$ actually produces the output for the entries. - * For an entry with an unknown type, it executes the function - * default.type. Thus you should define (before the READ command) - * one function for each standard entry type as well as a - * default.type function. - */ - buildInFunctions.put("call.type$", context -> { - if (context == null) { - throw new VMException("Call.type$ can only be called from within a context (ITERATE or REVERSE)."); - } - VM.this.execute(context.entry.getType().getName(), context); - }); - - buildInFunctions.put("change.case$", new ChangeCaseFunction(this)); - - /* - * Pops the top (string) literal, makes sure it's a single - * character, converts it to the corresponding ASCII integer, and - * pushes this integer. - */ - buildInFunctions.put("chr.to.int$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation chr.to.int$"); - } - Object o1 = stack.pop(); - - if (!((o1 instanceof String) && (((String) o1).length() == 1))) { - throw new VMException("Can only perform chr.to.int$ on string with length 1"); - } - - String s = (String) o1; - - stack.push((int) s.charAt(0)); - }); - - /* - * Pushes the string that was the \cite-command argument for this - * entry. - */ - buildInFunctions.put("cite$", context -> { - if (context == null) { - throw new VMException("Must have an entry to cite$"); - } - stack.push(context.entry.getCitationKey().orElse(null)); - }); - - /* - * Pops the top literal from the stack and pushes two copies of it. - */ - buildInFunctions.put("duplicate$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation duplicate$"); - } - Object o1 = stack.pop(); - - stack.push(o1); - stack.push(o1); - }); - - /* - * Pops the top literal and pushes the integer 1 if it's a missing - * field or a string having no non-white-space characters, 0 - * otherwise. - */ - buildInFunctions.put("empty$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation empty$"); - } - Object o1 = stack.pop(); - - if (o1 == null) { - stack.push(VM.TRUE); - return; - } - - if (!(o1 instanceof String)) { - throw new VMException("Operand does not match function empty$"); - } - - String s = (String) o1; - - stack.push("".equals(s.trim()) ? VM.TRUE : VM.FALSE); - }); - - buildInFunctions.put("format.name$", new FormatNameFunction(this)); - - /* - * Pops the top three literals (they are two function literals and - * an integer literal, in that order); if the integer is greater - * than 0, it executes the second literal, else it executes the - * first. - */ - buildInFunctions.put("if$", context -> { - if (stack.size() < 3) { - throw new VMException("Not enough operands on stack for operation ="); - } - Object f1 = stack.pop(); - Object f2 = stack.pop(); - Object i = stack.pop(); - - if (!((f1 instanceof Identifier) || (f1 instanceof Tree)) - && ((f2 instanceof Identifier) || (f2 instanceof Tree)) && (i instanceof Integer)) { - throw new VMException("Expecting two functions and an integer for if$."); - } - - if ((Integer) i > 0) { - VM.this.executeInContext(f2, context); - } else { - VM.this.executeInContext(f1, context); - } - }); - - /* - * Pops the top (integer) literal, interpreted as the ASCII integer - * value of a single character, converts it to the corresponding - * single-character string, and pushes this string. - */ - buildInFunctions.put("int.to.chr$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation int.to.chr$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof Integer)) { - throw new VMException("Can only perform operation int.to.chr$ on an Integer"); - } - - Integer i = (Integer) o1; - - stack.push(String.valueOf((char) i.intValue())); - }); - - /* - * Pops the top (integer) literal, converts it to its (unique) - * string equivalent, and pushes this string. - */ - buildInFunctions.put("int.to.str$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation int.to.str$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof Integer)) { - throw new VMException("Can only transform an integer to an string using int.to.str$"); - } - - stack.push(o1.toString()); - }); - - /* - * Pops the top literal and pushes the integer 1 if it's a missing - * field, 0 otherwise. - */ - buildInFunctions.put("missing$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation missing$"); - } - Object o1 = stack.pop(); - - if (o1 == null) { - stack.push(VM.TRUE); - return; - } - - if (!(o1 instanceof String)) { - warn("Not a string or missing field in operation missing$"); - stack.push(VM.TRUE); - return; - } - - stack.push(VM.FALSE); - }); - - /* - * Writes onto the bbl file what is accumulated in the output buffer. - * It writes a blank line if and only if the output buffer is empty. - * Since write$ does reasonable line breaking, you should use this - * function only when you want a blank line or an explicit line - * break. - */ - buildInFunctions.put("newline$", context -> VM.this.bbl.append('\n')); - - /* - * Pops the top (string) literal and pushes the number of names the - * string represents one plus the number of occurrences of the - * substring "and" (ignoring case differences) surrounded by - * non-null white-space at the top brace level. - */ - buildInFunctions.put("num.names$", context -> { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation num.names$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - throw new VMException("Need a string at the top of the stack for num.names$"); - } - String s = (String) o1; - - stack.push(AuthorList.parse(s).getNumberOfAuthors()); - }); - - /* - * Pops the top of the stack but doesn't print it; this gets rid of - * an unwanted stack literal. - */ - buildInFunctions.put("pop$", context -> stack.pop()); - - /* - * The |built_in| function {\.{preamble\$}} pushes onto the stack - * the concatenation of all the \.{preamble} strings read from the - * database files. (or the empty string if there where none) - * - * @PREAMBLE strings read from the database files. - */ - buildInFunctions.put("preamble$", context -> { - stack.push(preamble); - }); - - /* - * Pops the top (string) literal, removes nonalphanumeric characters - * except for white-space characters and hyphens and ties (these all get - * converted to a space), removes certain alphabetic characters - * contained in the control sequences associated with a \special - * character", and pushes the resulting string. - */ - buildInFunctions.put("purify$", new PurifyFunction(this)); - - /* - * Pushes the string consisting of the double-quote character. - */ - buildInFunctions.put("quote$", context -> stack.push("\"")); - - /* - * Is a no-op. - */ - buildInFunctions.put("skip$", context -> { - // Nothing to do! Yeah! - }); - - /* - * Pops and prints the whole stack; it's meant to be used for style - * designers while debugging. - */ - buildInFunctions.put("stack$", context -> { - while (!stack.empty()) { - LOGGER.debug("Stack entry {}", stack.pop()); - } - }); - - /* - * Pops the top three literals (they are the two integers literals - * len and start, and a string literal, in that order). It pushes - * the substring of the (at most) len consecutive characters - * starting at the startth character (assuming 1-based indexing) if - * start is positive, and ending at the start-th character - * (including) from the end if start is negative (where the first - * character from the end is the last character). - */ - buildInFunctions.put("substring$", context -> substringFunction()); - - /* - * Swaps the top two literals on the stack. text.length$ Pops the - * top (string) literal, and pushes the number of text characters - * it contains, where an accented character (more precisely, a - * \special character", defined in Section 4) counts as a single - * text character, even if it's missing its matching right brace, - * and where braces don't count as text characters. - */ - buildInFunctions.put("swap$", context -> { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation swap$"); - } - Object f1 = stack.pop(); - Object f2 = stack.pop(); - - stack.push(f1); - stack.push(f2); - }); - - /* - * text.length$ Pops the top (string) literal, and pushes the number - * of text characters it contains, where an accented character (more - * precisely, a "special character", defined in Section 4) counts as - * a single text character, even if it's missing its matching right - * brace, and where braces don't count as text characters. - * - * From BibTeXing: For the purposes of counting letters in labels, - * BibTEX considers everything contained inside the braces as a - * single letter. - */ - buildInFunctions.put("text.length$", context -> textLengthFunction()); - - /* - * Pops the top two literals (the integer literal len and a string - * literal, in that order). It pushes the substring of the (at most) len - * consecutive text characters starting from the beginning of the - * string. This function is similar to substring$, but this one - * considers a \special character", even if it's missing its matching - * right brace, to be a single text character (rather than however many - * ASCII characters it actually comprises), and this function doesn't - * consider braces to be text characters; furthermore, this function - * appends any needed matching right braces. - */ - buildInFunctions.put("text.prefix$", new TextPrefixFunction(this)); - - /* - * Pops and prints the top of the stack to the log file. It's useful for debugging. - */ - buildInFunctions.put("top$", context -> LOGGER.debug("Stack entry {}", stack.pop())); - - /* - * Pushes the current entry's type (book, article, etc.), but pushes - * the null string if the type is either unknown or undefined. - */ - buildInFunctions.put("type$", context -> { - if (context == null) { - throw new VMException("type$ need a context."); - } - - stack.push(context.entry.getType().getName()); - }); - - /* - * Pops the top (string) literal and prints it following a warning - * message. This also increments a count of the number of warning - * messages issued. - */ - buildInFunctions.put("warning$", new BstFunction() { - int warning = 1; - - @Override - public void execute(BstEntry context) { - LOGGER.warn("Warning (#" + (warning++) + "): " + stack.pop()); - } - }); - - /* - * Pops the top two (function) literals, and keeps executing the - * second as long as the (integer) literal left on the stack by - * executing the first is greater than 0. - */ - buildInFunctions.put("while$", this::whileFunction); - - buildInFunctions.put("width$", new WidthFunction(this)); - - /* - * Pops the top (string) literal and writes it on the output buffer - * (which will result in stuff being written onto the bbl file when - * the buffer fills up). - */ - buildInFunctions.put("write$", context -> { - String s = (String) stack.pop(); - VM.this.bbl.append(s); - }); - } - - private void textLengthFunction() { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation text.length$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - throw new VMException("Can only perform operation on a string text.length$"); - } - - String s = (String) o1; - char[] c = s.toCharArray(); - int result = 0; - - // Comments from bibtex.web: - - // sp_ptr := str_start[pop_lit1]; - int i = 0; - - // sp_end := str_start[pop_lit1+1]; - int n = s.length(); - - // sp_brace_level := 0; - int braceLevel = 0; - - // while (sp_ptr < sp_end) do begin - while (i < n) { - // incr(sp_ptr); - i++; - // if (str_pool[sp_ptr-1] = left_brace) then - // begin - if (c[i - 1] == '{') { - // incr(sp_brace_level); - braceLevel++; - // if ((sp_brace_level = 1) and (sp_ptr < sp_end)) then - if ((braceLevel == 1) && (i < n)) { - // if (str_pool[sp_ptr] = backslash) then - // begin - if (c[i] == '\\') { - // incr(sp_ptr); {skip over the |backslash|} - i++; // skip over backslash - // while ((sp_ptr < sp_end) and (sp_brace_level - // > 0)) do begin - while ((i < n) && (braceLevel > 0)) { - // if (str_pool[sp_ptr] = right_brace) then - if (c[i] == '}') { - // decr(sp_brace_level) - braceLevel--; - } else if (c[i] == '{') { - // incr(sp_brace_level); - braceLevel++; - } - // incr(sp_ptr); - i++; - // end; - } - // incr(num_text_chars); - result++; - // end; - } - // end - } - - // else if (str_pool[sp_ptr-1] = right_brace) then - // begin - } else if (c[i - 1] == '}') { - // if (sp_brace_level > 0) then - if (braceLevel > 0) { - // decr(sp_brace_level); - braceLevel--; - // end - } - } else { // else - // incr(num_text_chars); - result++; - } - } - stack.push(result); - } - - private void whileFunction(BstEntry context) { - if (stack.size() < 2) { - throw new VMException("Not enough operands on stack for operation while$"); - } - Object f2 = stack.pop(); - Object f1 = stack.pop(); - - if (!((f1 instanceof Identifier) || (f1 instanceof Tree)) - && ((f2 instanceof Identifier) || (f2 instanceof Tree))) { - throw new VMException("Expecting two functions for while$."); - } - - do { - VM.this.executeInContext(f1, context); - - Object i = stack.pop(); - if (!(i instanceof Integer)) { - throw new VMException("First parameter to while has to return an integer but was " + i); - } - if ((Integer) i <= 0) { - break; - } - VM.this.executeInContext(f2, context); - } while (true); - } - - private void substringFunction() { - if (stack.size() < 3) { - throw new VMException("Not enough operands on stack for operation substring$"); - } - Object o1 = stack.pop(); - Object o2 = stack.pop(); - Object o3 = stack.pop(); - - if (!((o1 instanceof Integer) && (o2 instanceof Integer) && (o3 instanceof String))) { - throw new VMException("Expecting two integers and a string for substring$"); - } - - Integer len = (Integer) o1; - Integer start = (Integer) o2; - - int lenI = len; - int startI = start; - - if (lenI > (Integer.MAX_VALUE / 2)) { - lenI = Integer.MAX_VALUE / 2; - } - - if (startI > (Integer.MAX_VALUE / 2)) { - startI = Integer.MAX_VALUE / 2; - } - - if (startI < (Integer.MIN_VALUE / 2)) { - startI = -Integer.MIN_VALUE / 2; - } - - String s = (String) o3; - - if (startI < 0) { - startI += s.length() + 1; - startI = Math.max(1, (startI + 1) - lenI); - } - stack.push(s.substring(startI - 1, Math.min((startI - 1) + lenI, s.length()))); - } - - private void addPeriodFunction() { - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation add.period$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - throw new VMException("Can only add a period to a string for add.period$"); - } - - String s = (String) o1; - Matcher m = ADD_PERIOD_PATTERN.matcher(s); - - if (m.find()) { - StringBuilder sb = new StringBuilder(); - m.appendReplacement(sb, m.group(1)); - sb.append('.'); - String group2 = m.group(2); - if (group2 != null) { - sb.append(m.group(2)); - } - stack.push(sb.toString()); - } else { - stack.push(s); - } - } - - private static CommonTree charStream2CommonTree(CharStream bst) throws RecognitionException { - BstLexer lex = new BstLexer(bst); - CommonTokenStream tokens = new CommonTokenStream(lex); - BstParser parser = new BstParser(tokens); - BstParser.program_return r = parser.program(); - return (CommonTree) r.getTree(); - } - - private boolean assign(BstEntry context, Object o1, Object o2) { - if (!(o1 instanceof Identifier) || !((o2 instanceof String) || (o2 instanceof Integer))) { - throw new VMException("Invalid parameters"); - } - - String name = ((Identifier) o1).getName(); - - if (o2 instanceof String) { - if ((context != null) && context.localStrings.containsKey(name)) { - context.localStrings.put(name, (String) o2); - return true; - } - - if (strings.containsKey(name)) { - strings.put(name, (String) o2); - return true; - } - return false; - } - - if ((context != null) && context.localIntegers.containsKey(name)) { - context.localIntegers.put(name, (Integer) o2); - return true; - } - - if (integers.containsKey(name)) { - integers.put(name, (Integer) o2); - return true; - } - return false; - } - - public String run(BibDatabase db) { - preamble = db.getPreamble().orElse(""); - return run(db.getEntries()); - } - - public String run(Collection bibtex) { - return this.run(bibtex, null); - } - - /** - * Transforms the given list of BibEntries to a rendered list of references using the underlying bst file - * - * @param bibEntries list of entries to convert - * @param bibDatabase (may be null) the bibDatabase used for resolving strings / crossref - * @return list of references in plain text form - */ - public String run(Collection bibEntries, BibDatabase bibDatabase) { - Objects.requireNonNull(bibEntries); - - // Reset - bbl = new StringBuilder(); - - strings = new HashMap<>(); - - integers = new HashMap<>(); - integers.put("entry.max$", Integer.MAX_VALUE); - integers.put("global.max$", Integer.MAX_VALUE); - - functions = new HashMap<>(); - functions.putAll(buildInFunctions); - - stack = new Stack<>(); - - // Create entries - entries = new ArrayList<>(bibEntries.size()); - for (BibEntry entry : bibEntries) { - entries.add(new BstEntry(entry)); - } - - // Go - for (int i = 0; i < tree.getChildCount(); i++) { - Tree child = tree.getChild(i); - switch (child.getType()) { - case BstParser.STRINGS: - strings(child); - break; - case BstParser.INTEGERS: - integers(child); - break; - case BstParser.FUNCTION: - function(child); - break; - case BstParser.EXECUTE: - execute(child); - break; - case BstParser.SORT: - sort(); - break; - case BstParser.ITERATE: - iterate(child); - break; - case BstParser.REVERSE: - reverse(child); - break; - case BstParser.ENTRY: - entry(child); - break; - case BstParser.READ: - read(bibDatabase); - break; - case BstParser.MACRO: - macro(child); - break; - default: - LOGGER.info("Unknown type: {}", child.getType()); - break; - } - } - - return bbl.toString(); - } - - /** - * Dredges up from the database file the field values for each entry in the list. It has no arguments. If a database - * entry doesn't have a value for a field (and probably no database entry will have a value for every field), that - * field variable is marked as missing for the entry. - * - * We use null for the missing entry designator. - */ - private void read(BibDatabase bibDatabase) { - FieldWriter fieldWriter = new FieldWriter(new FieldWriterPreferences(true, List.of(StandardField.MONTH), new FieldContentFormatterPreferences())); - for (BstEntry e : entries) { - for (Map.Entry mEntry : e.fields.entrySet()) { - Field field = FieldFactory.parseField(mEntry.getKey()); - String fieldValue = e.entry.getResolvedFieldOrAlias(field, bibDatabase) - .map(content -> { - try { - String result = fieldWriter.write(field, content); - if (result.startsWith("{")) { - // Strip enclosing {} from the output - return result.substring(1, result.length() - 1); - } - if (field == StandardField.MONTH) { - // We don't have the internal BibTeX strings at hand. - // We nevertheless want to have the full month name. - // Thus, we lookup the full month name here. - return Month.parse(result) - .map(month -> month.getFullName()) - .orElse(result); - } - return result; - } catch (InvalidFieldValueException invalidFieldValueException) { - // in case there is something wrong with the content, just return the content itself - return content; - } - }) - .orElse(null); - mEntry.setValue(fieldValue); - } - } - - for (BstEntry e : entries) { - if (!e.fields.containsKey(StandardField.CROSSREF.getName())) { - e.fields.put(StandardField.CROSSREF.getName(), null); - } - } - } - - /** - * Defines a string macro. It has two arguments; the first is the macro's name, which is treated like any other - * variable or function name, and the second is its definition, which must be double-quote-delimited. You must have - * one for each three-letter month abbreviation; in addition, you should have one for common journal names. The - * user's database may override any definition you define using this command. If you want to define a string the - * user can't touch, use the FUNCTION command, which has a compatible syntax. - */ - private void macro(Tree child) { - String name = child.getChild(0).getText(); - String replacement = child.getChild(1).getText(); - functions.put(name, new MacroFunction(replacement)); - } - - public class MacroFunction implements BstFunction { - - private final String replacement; - - public MacroFunction(String replacement) { - this.replacement = replacement; - } - - @Override - public void execute(BstEntry context) { - VM.this.push(replacement); - } - } - - /** - * Declares the fields and entry variables. It has three arguments, each a (possibly empty) list of variable names. - * The three lists are of: fields, integer entry variables, and string entry variables. There is an additional field - * that BibTEX automatically declares, crossref, used for cross referencing. And there is an additional string entry - * variable automatically declared, sort.key$, used by the SORT command. Each of these variables has a value for - * each entry on the list. - */ - private void entry(Tree child) { - // Fields first - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - - for (BstEntry entry : entries) { - entry.fields.put(name, null); - } - } - - // Integers - t = child.getChild(1); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - - for (BstEntry entry : entries) { - entry.localIntegers.put(name, 0); - } - } - // Strings - t = child.getChild(2); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - for (BstEntry entry : entries) { - entry.localStrings.put(name, null); - } - } - for (BstEntry entry : entries) { - entry.localStrings.put("sort.key$", null); - } - } - - private void reverse(Tree child) { - BstFunction f = functions.get(child.getChild(0).getText()); - - ListIterator i = entries.listIterator(entries.size()); - while (i.hasPrevious()) { - f.execute(i.previous()); - } - } - - private void iterate(Tree child) { - BstFunction f = functions.get(child.getChild(0).getText()); - - for (BstEntry entry : entries) { - f.execute(entry); - } - } - - /** - * Sorts the entry list using the values of the string entry variable sort.key$. It has no arguments. - */ - private void sort() { - entries.sort(Comparator.comparing(o -> (o.localStrings.get("sort.key$")))); - } - - private void executeInContext(Object o, BstEntry context) { - if (o instanceof Tree) { - Tree t = (Tree) o; - new StackFunction(t).execute(context); - } else if (o instanceof Identifier) { - execute(((Identifier) o).getName(), context); - } - } - - private void execute(Tree child) { - execute(child.getChild(0).getText(), null); - } - - public class StackFunction implements BstFunction { - - private final Tree localTree; - - public StackFunction(Tree stack) { - localTree = stack; - } - - public Tree getTree() { - return localTree; - } - - @Override - public void execute(BstEntry context) { - for (int i = 0; i < localTree.getChildCount(); i++) { - Tree c = localTree.getChild(i); - try { - - switch (c.getType()) { - case BstParser.STRING: - String s = c.getText(); - push(s.substring(1, s.length() - 1)); - break; - case BstParser.INTEGER: - push(Integer.parseInt(c.getText().substring(1))); - break; - case BstParser.QUOTED: - push(new Identifier(c.getText().substring(1))); - break; - case BstParser.STACK: - push(c); - break; - default: - VM.this.execute(c.getText(), context); - break; - } - } catch (VMException e) { - if (file == null) { - LOGGER.error("ERROR " + e.getMessage() + " (" + c.getLine() + ")"); - } else { - LOGGER.error("ERROR " + e.getMessage() + " (" + file.getPath() + ":" - + c.getLine() + ")"); - } - throw e; - } - } - } - } - - private void push(Tree t) { - stack.push(t); - } - - private void execute(String name, BstEntry context) { - if (context != null) { - if (context.fields.containsKey(name)) { - stack.push(context.fields.get(name)); - return; - } - if (context.localStrings.containsKey(name)) { - stack.push(context.localStrings.get(name)); - return; - } - if (context.localIntegers.containsKey(name)) { - stack.push(context.localIntegers.get(name)); - return; - } - } - if (strings.containsKey(name)) { - stack.push(strings.get(name)); - return; - } - if (integers.containsKey(name)) { - stack.push(integers.get(name)); - return; - } - - if (functions.containsKey(name)) { - // OK to have a null context - functions.get(name).execute(context); - return; - } - - throw new VMException("No matching identifier found: " + name); - } - - private void function(Tree child) { - String name = child.getChild(0).getText(); - Tree localStack = child.getChild(1); - functions.put(name, new StackFunction(localStack)); - } - - /** - * Declares global integer variables. It has one argument, a list of variable names. There are two such - * automatically-declared variables, entry.max$ and global.max$, used for limiting the lengths of string vari- - * ables. You may have any number of these commands, but a variable's declaration must precede its use. - */ - private void integers(Tree child) { - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - integers.put(name, 0); - } - } - - /** - * Declares global string variables. It has one argument, a list of variable names. You may have any number of these - * commands, but a variable's declaration must precede its use. - * - * @param child - */ - private void strings(Tree child) { - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - strings.put(name, null); - } - } - - public static class BstEntry { - - public final BibEntry entry; - - public final Map localStrings = new HashMap<>(); - - // keys filled by org.jabref.logic.bst.VM.entry based on the contents of the bst file - public final Map fields = new HashMap<>(); - - public final Map localIntegers = new HashMap<>(); - - public BstEntry(BibEntry e) { - this.entry = e; - } - } - - private void push(Integer integer) { - stack.push(integer); - } - - private void push(String string) { - stack.push(string); - } - - private void push(Identifier identifier) { - stack.push(identifier); - } - - public Map getStrings() { - return strings; - } - - public Map getIntegers() { - return integers; - } - - public List getEntries() { - return entries; - } - - public Map getFunctions() { - return functions; - } - - public Stack getStack() { - return stack; - } - - @Override - public void warn(String string) { - LOGGER.warn(string); - } -} diff --git a/src/main/java/org/jabref/logic/bst/VMException.java b/src/main/java/org/jabref/logic/bst/VMException.java deleted file mode 100644 index f46c5cd5277..00000000000 --- a/src/main/java/org/jabref/logic/bst/VMException.java +++ /dev/null @@ -1,8 +0,0 @@ -package org.jabref.logic.bst; - -public class VMException extends RuntimeException { - - public VMException(String string) { - super(string); - } -} diff --git a/src/main/java/org/jabref/logic/bst/Warn.java b/src/main/java/org/jabref/logic/bst/Warn.java deleted file mode 100644 index 7a524ad9834..00000000000 --- a/src/main/java/org/jabref/logic/bst/Warn.java +++ /dev/null @@ -1,7 +0,0 @@ -package org.jabref.logic.bst; - -@FunctionalInterface -public interface Warn { - - void warn(String s); -} diff --git a/src/main/java/org/jabref/logic/bst/WidthFunction.java b/src/main/java/org/jabref/logic/bst/WidthFunction.java deleted file mode 100644 index 06784267bf0..00000000000 --- a/src/main/java/org/jabref/logic/bst/WidthFunction.java +++ /dev/null @@ -1,43 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Stack; - -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; - -/** - * The |built_in| function {\.{width\$}} pops the top (string) literal and - * pushes the integer that represents its width in units specified by the - * |char_width| array. This function takes the literal literally; that is, it - * assumes each character in the string is to be printed as is, regardless of - * whether the character has a special meaning to \TeX, except that special - * characters (even without their |right_brace|s) are handled specially. If the - * literal isn't a string, it complains and pushes~0. - * - */ -public class WidthFunction implements BstFunction { - - private final VM vm; - - public WidthFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation width$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - vm.warn("A string is needed for change.case$"); - stack.push(0); - return; - } - - stack.push(BibtexWidth.width((String) o1)); - } -} diff --git a/src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java b/src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java similarity index 83% rename from src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java rename to src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java index e05ce998e80..dd3e7c7b05c 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java +++ b/src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java @@ -1,4 +1,4 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; import java.util.Locale; import java.util.Optional; @@ -6,9 +6,9 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; -public final class BibtexCaseChanger { +public final class BstCaseChanger { - private static final Logger LOGGER = LoggerFactory.getLogger(BibtexCaseChanger.class); + private static final Logger LOGGER = LoggerFactory.getLogger(BstCaseChanger.class); // stores whether the char before the current char was a colon private boolean prevColon = true; @@ -16,7 +16,7 @@ public final class BibtexCaseChanger { // global variable to store the current brace level private int braceLevel; - public enum FORMAT_MODE { + public enum FormatMode { // First character and character after a ":" as upper case - everything else in lower case. Obey {}. TITLE_LOWERS('t'), @@ -40,7 +40,7 @@ public enum FORMAT_MODE { private final char asChar; - FORMAT_MODE(char asChar) { + FormatMode(char asChar) { this.asChar = asChar; } @@ -53,17 +53,21 @@ public char asChar() { * * @throws IllegalArgumentException if char is not 't', 'l', 'u' */ - public static FORMAT_MODE getFormatModeForBSTFormat(final char bstFormat) { - for (FORMAT_MODE mode : FORMAT_MODE.values()) { + public static FormatMode of(final char bstFormat) { + for (FormatMode mode : FormatMode.values()) { if (mode.asChar == bstFormat) { return mode; } } throw new IllegalArgumentException(); } + + public static FormatMode of(final String bstFormat) { + return of(bstFormat.toLowerCase(Locale.ROOT).charAt(0)); + } } - private BibtexCaseChanger() { + private BstCaseChanger() { } /** @@ -72,11 +76,11 @@ private BibtexCaseChanger() { * @param s the string to handle * @param format the format */ - public static String changeCase(String s, FORMAT_MODE format) { - return (new BibtexCaseChanger()).doChangeCase(s, format); + public static String changeCase(String s, FormatMode format) { + return (new BstCaseChanger()).doChangeCase(s, format); } - private String doChangeCase(String s, FORMAT_MODE format) { + private String doChangeCase(String s, FormatMode format) { char[] c = s.toCharArray(); StringBuilder sb = new StringBuilder(); @@ -93,7 +97,7 @@ private String doChangeCase(String s, FORMAT_MODE format) { i++; continue; } - if ((format == FORMAT_MODE.TITLE_LOWERS) && ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1])))) { + if ((format == FormatMode.TITLE_LOWERS) && ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1])))) { sb.append('{'); i++; prevColon = false; @@ -136,12 +140,9 @@ private String doChangeCase(String s, FORMAT_MODE format) { * is other stuff, too, between braces, but it doesn't try to do anything * special with |colon|s. * - * @param c * @param start the current position. It points to the opening brace - * @param format - * @return */ - private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MODE format) { + private int convertSpecialChar(StringBuilder sb, char[] c, int start, FormatMode format) { int i = start; sb.append(c[i]); @@ -152,7 +153,7 @@ private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MOD i++; // skip over the |backslash| - Optional s = BibtexCaseChanger.findSpecialChar(c, i); + Optional s = BstCaseChanger.findSpecialChar(c, i); if (s.isPresent()) { i = convertAccented(c, i, s.get(), sb, format); } @@ -174,14 +175,9 @@ private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MOD * up) and append the result to the stringBuffer, return the updated * position. * - * @param c - * @param start - * @param s - * @param sb - * @param format * @return the new position */ - private int convertAccented(char[] c, int start, String s, StringBuilder sb, FORMAT_MODE format) { + private int convertAccented(char[] c, int start, String s, StringBuilder sb, FormatMode format) { int pos = start; pos += s.length(); @@ -214,29 +210,27 @@ private int convertAccented(char[] c, int start, String s, StringBuilder sb, FOR return pos; } - private int convertNonControl(char[] c, int start, StringBuilder sb, FORMAT_MODE format) { + private int convertNonControl(char[] c, int start, StringBuilder sb, FormatMode format) { int pos = start; switch (format) { - case TITLE_LOWERS: - case ALL_LOWERS: + case TITLE_LOWERS, ALL_LOWERS -> { sb.append(Character.toLowerCase(c[pos])); pos++; - break; - case ALL_UPPERS: + } + case ALL_UPPERS -> { sb.append(Character.toUpperCase(c[pos])); pos++; - break; - default: - LOGGER.info("convertNonControl - Unknown format: " + format); - break; + } + default -> + LOGGER.info("convertNonControl - Unknown format: " + format); } return pos; } - private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, FORMAT_MODE format) { + private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, FormatMode format) { int i = start; switch (format) { - case TITLE_LOWERS: + case TITLE_LOWERS -> { if ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1]))) { sb.append(c[i]); } else { @@ -247,16 +241,13 @@ private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, } else if (!Character.isWhitespace(c[i])) { prevColon = false; } - break; - case ALL_LOWERS: - sb.append(Character.toLowerCase(c[i])); - break; - case ALL_UPPERS: - sb.append(Character.toUpperCase(c[i])); - break; - default: - LOGGER.info("convertCharIfBraceLevelIsZero - Unknown format: " + format); - break; + } + case ALL_LOWERS -> + sb.append(Character.toLowerCase(c[i])); + case ALL_UPPERS -> + sb.append(Character.toUpperCase(c[i])); + default -> + LOGGER.info("convertCharIfBraceLevelIsZero - Unknown format: " + format); } i++; return i; diff --git a/src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java b/src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java similarity index 76% rename from src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java rename to src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java index aecf571bdf4..c4986dc12dd 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java +++ b/src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java @@ -1,13 +1,17 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; import java.util.Arrays; import java.util.Locale; import java.util.Optional; import java.util.stream.Collectors; +import org.jabref.logic.bst.BstVMException; import org.jabref.model.entry.Author; import org.jabref.model.entry.AuthorList; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + /** * From Bibtex: * @@ -24,9 +28,10 @@ * Sounds easy - is a nightmare... X-( * */ -public class BibtexNameFormatter { +public class BstNameFormatter { + private static final Logger LOGGER = LoggerFactory.getLogger(BstNameFormatter.class); - private BibtexNameFormatter() { + private BstNameFormatter() { } /** @@ -35,23 +40,18 @@ private BibtexNameFormatter() { * @param authorsNameList The string from an author field * @param whichName index of the list, starting with 1 * @param formatString TODO - * @param warn collects the warnings, may-be-null - * @return */ - public static String formatName(String authorsNameList, int whichName, String formatString, Warn warn) { + public static String formatName(String authorsNameList, int whichName, String formatString) { AuthorList al = AuthorList.parse(authorsNameList); if ((whichName < 1) && (whichName > al.getNumberOfAuthors())) { - warn.warn("AuthorList " + authorsNameList + " does not contain an author with number " + whichName); + LOGGER.warn("AuthorList {} does not contain an author with number {}", authorsNameList, whichName); return ""; } - return BibtexNameFormatter.formatName(al.getAuthor(whichName - 1), formatString, warn); + return BstNameFormatter.formatName(al.getAuthor(whichName - 1), formatString); } - /** - * @param warn collects the warnings, may-be-null - */ - public static String formatName(Author author, String format, Warn warn) { + public static String formatName(Author author, String format) { StringBuilder sb = new StringBuilder(); char[] c = format.toCharArray(); @@ -81,11 +81,7 @@ public static String formatName(Author author, String format, Warn warn) { } if ((braceLevel == 1) && Character.isLetter(c[i])) { if ("fvlj".indexOf(c[i]) == -1) { - if (warn != null) { - warn.warn( - "Format string in format.name$ may only contain fvlj on brace level 1 in group " - + group + ": " + format); - } + LOGGER.warn("Format string in format.name$ may only contain fvlj on brace level 1 in group {}: {}", group, format); } else { level1Chars.append(c[i]); } @@ -99,31 +95,26 @@ public static String formatName(Author author, String format, Warn warn) { continue; } - if ((control.length() > 2) && (warn != null)) { - warn.warn("Format string in format.name$ may only be one or two character long on brace level 1 in group " + group + ": " + format); + if ((control.length() > 2)) { + LOGGER.warn("Format string in format.name$ may only be one or two character long on brace level 1 in group {}: {}", group, format); } char type = control.charAt(0); - Optional tokenS; - switch (type) { - case 'f': - tokenS = author.getFirst(); - break; - case 'v': - tokenS = author.getVon(); - break; - case 'l': - tokenS = author.getLast(); - break; - case 'j': - tokenS = author.getJr(); - break; - default: - throw new VMException("Internal error"); - } - - if (!tokenS.isPresent()) { + Optional tokenS = switch (type) { + case 'f' -> + author.getFirst(); + case 'v' -> + author.getVon(); + case 'l' -> + author.getLast(); + case 'j' -> + author.getJr(); + default -> + throw new BstVMException("Internal error"); + }; + + if (tokenS.isEmpty()) { i++; continue; } @@ -135,9 +126,7 @@ public static String formatName(Author author, String format, Warn warn) { if (control.charAt(1) == control.charAt(0)) { abbreviateThatIsSingleLetter = false; } else { - if (warn != null) { - warn.warn("Format string in format.name$ may only contain one type of vlfj on brace level 1 in group " + group + ": " + format); - } + LOGGER.warn("Format string in format.name$ may only contain one type of vlfj on brace level 1 in group {}: {}", group, format); } } @@ -162,7 +151,7 @@ public static String formatName(Author author, String format, Warn warn) { } if (((j + 1) < d.length) && (d[j + 1] == '{')) { StringBuilder interTokenSb = new StringBuilder(); - j = BibtexNameFormatter.consumeToMatchingBrace(interTokenSb, d, j + 1); + j = BstNameFormatter.consumeToMatchingBrace(interTokenSb, d, j + 1); interToken = interTokenSb.substring(1, interTokenSb.length() - 1); } @@ -171,7 +160,7 @@ public static String formatName(Author author, String format, Warn warn) { if (abbreviateThatIsSingleLetter) { String[] dashes = token.split("-"); - token = Arrays.asList(dashes).stream().map(BibtexNameFormatter::getFirstCharOfString) + token = Arrays.stream(dashes).map(BstNameFormatter::getFirstCharOfString) .collect(Collectors.joining(".-")); } @@ -187,7 +176,7 @@ public static String formatName(Author author, String format, Warn warn) { // No clue what this means (What the hell are tokens anyway??? // if (lex_class[name_sep_char[cur_token]] = sep_char) then // append_ex_buf_char_and_check (name_sep_char[cur_token]) - if ((k == (tokens.length - 2)) || (BibtexNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 3) < 3)) { + if ((k == (tokens.length - 2)) || (BstNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 3) < 3)) { sb.append('~'); } else { sb.append(' '); @@ -212,7 +201,7 @@ public static String formatName(Author author, String format, Warn warn) { if (sb.length() > 0) { boolean noDisTie = false; if ((sb.charAt(sb.length() - 1) == '~') && - ((BibtexNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 4) >= 4) || + ((BstNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 4) >= 4) || ((sb.length() > 1) && (noDisTie = sb.charAt(sb.length() - 2) == '~')))) { sb.deleteCharAt(sb.length() - 1); if (!noDisTie) { @@ -221,16 +210,14 @@ public static String formatName(Author author, String format, Warn warn) { } } } else if (c[i] == '}') { - if (warn != null) { - warn.warn("Unmatched brace in format string: " + format); - } + LOGGER.warn("Unmatched brace in format string: {}", format); } else { sb.append(c[i]); // verbatim } i++; } - if ((braceLevel != 0) && (warn != null)) { - warn.warn("Unbalanced brace in format string for nameFormat: " + format); + if ((braceLevel != 0)) { + LOGGER.warn("Unbalanced brace in format string for nameFormat: {}", format); } return sb.toString(); @@ -268,7 +255,7 @@ public static String getFirstCharOfString(String s) { } if ((c[i] == '{') && ((i + 1) < c.length) && (c[i + 1] == '\\')) { StringBuilder sb = new StringBuilder(); - BibtexNameFormatter.consumeToMatchingBrace(sb, c, i); + BstNameFormatter.consumeToMatchingBrace(sb, c, i); return sb.toString(); } } diff --git a/src/main/java/org/jabref/logic/bst/BibtexPurify.java b/src/main/java/org/jabref/logic/bst/util/BstPurifier.java similarity index 77% rename from src/main/java/org/jabref/logic/bst/BibtexPurify.java rename to src/main/java/org/jabref/logic/bst/util/BstPurifier.java index 08cb14db207..90818a0749f 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexPurify.java +++ b/src/main/java/org/jabref/logic/bst/util/BstPurifier.java @@ -1,4 +1,7 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * @@ -10,17 +13,13 @@ * pushes the null string. * */ -public class BibtexPurify { +public class BstPurifier { + private static final Logger LOGGER = LoggerFactory.getLogger(BstPurifier.class); - private BibtexPurify() { + private BstPurifier() { } - /** - * @param toPurify - * @param warn may-be-null - * @return - */ - public static String purify(String toPurify, Warn warn) { + public static String purify(String toPurify) { StringBuilder sb = new StringBuilder(); char[] cs = toPurify.toCharArray(); @@ -41,7 +40,7 @@ public static String purify(String toPurify, Warn warn) { i++; // skip brace while ((i < n) && (braceLevel > 0)) { i++; // skip backslash - BibtexCaseChanger.findSpecialChar(cs, i).ifPresent(sb::append); + BstCaseChanger.findSpecialChar(cs, i).ifPresent(sb::append); while ((i < n) && Character.isLetter(cs[i])) { i++; @@ -63,15 +62,13 @@ public static String purify(String toPurify, Warn warn) { if (braceLevel > 0) { braceLevel--; } else { - if (warn != null) { - warn.warn("Unbalanced brace in string for purify$: " + toPurify); - } + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPurify); } } i++; } - if ((braceLevel != 0) && (warn != null)) { - warn.warn("Unbalanced brace in string for purify$: " + toPurify); + if ((braceLevel != 0)) { + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPurify); } return sb.toString(); diff --git a/src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java b/src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java similarity index 84% rename from src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java rename to src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java index 721838e29fe..31cbc71de9b 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java +++ b/src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java @@ -1,4 +1,7 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * The |built_in| function {\.{text.prefix\$}} pops the top two literals (the @@ -14,15 +17,13 @@ * complains and pushes the null string. * */ -public class BibtexTextPrefix { +public class BstTextPrefixer { + private static final Logger LOGGER = LoggerFactory.getLogger(BstTextPrefixer.class); - private BibtexTextPrefix() { + private BstTextPrefixer() { } - /** - * @param warn may-be-null - */ - public static String textPrefix(int inNumOfChars, String toPrefix, Warn warn) { + public static String textPrefix(int inNumOfChars, String toPrefix) { int numOfChars = inNumOfChars; StringBuilder sb = new StringBuilder(); @@ -53,15 +54,13 @@ public static String textPrefix(int inNumOfChars, String toPrefix, Warn warn) { if (braceLevel > 0) { braceLevel--; } else { - if (warn != null) { - warn.warn("Unbalanced brace in string for purify$: " + toPrefix); - } + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPrefix); } } else { numOfChars--; } } - sb.append(toPrefix.substring(0, i)); + sb.append(toPrefix, 0, i); while (braceLevel > 0) { sb.append('}'); braceLevel--; diff --git a/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java b/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java new file mode 100644 index 00000000000..c286fe1497b --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java @@ -0,0 +1,241 @@ +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * + * The |built_in| function {\.{purify\$}} pops the top (string) literal, removes + * nonalphanumeric characters except for |white_space| and |sep_char| characters + * (these get converted to a |space|) and removes certain alphabetic characters + * contained in the control sequences associated with a special character, and + * pushes the resulting string. If the literal isn't a string, it complains and + * pushes the null string. + * + */ +public class BstWidthCalculator { + + private static final Logger LOGGER = LoggerFactory.getLogger(BstWidthCalculator.class); + + /* + * Quoted from Bibtex: + * + * Now we initialize the system-dependent |char_width| array, for which + * |space| is the only |white_space| character given a nonzero printing + * width. The widths here are taken from Stanford's June~'87 $cmr10$~font + * and represent hundredths of a point (rounded), but since they're used + * only for relative comparisons, the units have no meaning. + */ + + private static int[] widths; + + static { + if (BstWidthCalculator.widths == null) { + BstWidthCalculator.widths = new int[128]; + + for (int i = 0; i < 128; i++) { + BstWidthCalculator.widths[i] = 0; + } + BstWidthCalculator.widths[32] = 278; + BstWidthCalculator.widths[33] = 278; + BstWidthCalculator.widths[34] = 500; + BstWidthCalculator.widths[35] = 833; + BstWidthCalculator.widths[36] = 500; + BstWidthCalculator.widths[37] = 833; + BstWidthCalculator.widths[38] = 778; + BstWidthCalculator.widths[39] = 278; + BstWidthCalculator.widths[40] = 389; + BstWidthCalculator.widths[41] = 389; + BstWidthCalculator.widths[42] = 500; + BstWidthCalculator.widths[43] = 778; + BstWidthCalculator.widths[44] = 278; + BstWidthCalculator.widths[45] = 333; + BstWidthCalculator.widths[46] = 278; + BstWidthCalculator.widths[47] = 500; + BstWidthCalculator.widths[48] = 500; + BstWidthCalculator.widths[49] = 500; + BstWidthCalculator.widths[50] = 500; + BstWidthCalculator.widths[51] = 500; + BstWidthCalculator.widths[52] = 500; + BstWidthCalculator.widths[53] = 500; + BstWidthCalculator.widths[54] = 500; + BstWidthCalculator.widths[55] = 500; + BstWidthCalculator.widths[56] = 500; + BstWidthCalculator.widths[57] = 500; + BstWidthCalculator.widths[58] = 278; + BstWidthCalculator.widths[59] = 278; + BstWidthCalculator.widths[60] = 278; + BstWidthCalculator.widths[61] = 778; + BstWidthCalculator.widths[62] = 472; + BstWidthCalculator.widths[63] = 472; + BstWidthCalculator.widths[64] = 778; + BstWidthCalculator.widths[65] = 750; + BstWidthCalculator.widths[66] = 708; + BstWidthCalculator.widths[67] = 722; + BstWidthCalculator.widths[68] = 764; + BstWidthCalculator.widths[69] = 681; + BstWidthCalculator.widths[70] = 653; + BstWidthCalculator.widths[71] = 785; + BstWidthCalculator.widths[72] = 750; + BstWidthCalculator.widths[73] = 361; + BstWidthCalculator.widths[74] = 514; + BstWidthCalculator.widths[75] = 778; + BstWidthCalculator.widths[76] = 625; + BstWidthCalculator.widths[77] = 917; + BstWidthCalculator.widths[78] = 750; + BstWidthCalculator.widths[79] = 778; + BstWidthCalculator.widths[80] = 681; + BstWidthCalculator.widths[81] = 778; + BstWidthCalculator.widths[82] = 736; + BstWidthCalculator.widths[83] = 556; + BstWidthCalculator.widths[84] = 722; + BstWidthCalculator.widths[85] = 750; + BstWidthCalculator.widths[86] = 750; + BstWidthCalculator.widths[87] = 1028; + BstWidthCalculator.widths[88] = 750; + BstWidthCalculator.widths[89] = 750; + BstWidthCalculator.widths[90] = 611; + BstWidthCalculator.widths[91] = 278; + BstWidthCalculator.widths[92] = 500; + BstWidthCalculator.widths[93] = 278; + BstWidthCalculator.widths[94] = 500; + BstWidthCalculator.widths[95] = 278; + BstWidthCalculator.widths[96] = 278; + BstWidthCalculator.widths[97] = 500; + BstWidthCalculator.widths[98] = 556; + BstWidthCalculator.widths[99] = 444; + BstWidthCalculator.widths[100] = 556; + BstWidthCalculator.widths[101] = 444; + BstWidthCalculator.widths[102] = 306; + BstWidthCalculator.widths[103] = 500; + BstWidthCalculator.widths[104] = 556; + BstWidthCalculator.widths[105] = 278; + BstWidthCalculator.widths[106] = 306; + BstWidthCalculator.widths[107] = 528; + BstWidthCalculator.widths[108] = 278; + BstWidthCalculator.widths[109] = 833; + BstWidthCalculator.widths[110] = 556; + BstWidthCalculator.widths[111] = 500; + BstWidthCalculator.widths[112] = 556; + BstWidthCalculator.widths[113] = 528; + BstWidthCalculator.widths[114] = 392; + BstWidthCalculator.widths[115] = 394; + BstWidthCalculator.widths[116] = 389; + BstWidthCalculator.widths[117] = 556; + BstWidthCalculator.widths[118] = 528; + BstWidthCalculator.widths[119] = 722; + BstWidthCalculator.widths[120] = 528; + BstWidthCalculator.widths[121] = 528; + BstWidthCalculator.widths[122] = 444; + BstWidthCalculator.widths[123] = 500; + BstWidthCalculator.widths[124] = 1000; + BstWidthCalculator.widths[125] = 500; + BstWidthCalculator.widths[126] = 500; + } + } + + private BstWidthCalculator() { + } + + private static int getSpecialCharWidth(char[] c, int pos) { + if ((pos + 1) < c.length) { + if ((c[pos] == 'o') && (c[pos + 1] == 'e')) { + return 778; + } + if ((c[pos] == 'O') && (c[pos + 1] == 'E')) { + return 1014; + } + if ((c[pos] == 'a') && (c[pos + 1] == 'e')) { + return 722; + } + if ((c[pos] == 'A') && (c[pos + 1] == 'E')) { + return 903; + } + if ((c[pos] == 's') && (c[pos + 1] == 's')) { + return 500; + } + } + return BstWidthCalculator.getCharWidth(c[pos]); + } + + public static int getCharWidth(char c) { + if ((c >= 0) && (c < 128)) { + return BstWidthCalculator.widths[c]; + } else { + return 0; + } + } + + public static int width(String toMeasure) { + /* + * From Bibtex: We use the natural width for all but special characters, + * and we complain if the string isn't brace-balanced. + */ + + int i = 0; + int n = toMeasure.length(); + int braceLevel = 0; + char[] c = toMeasure.toCharArray(); + int result = 0; + + /* + * From Bibtex: + * + * We use the natural widths of all characters except that some + * characters have no width: braces, control sequences (except for the + * usual 13 accented and foreign characters, whose widths are given in + * the next module), and |white_space| following control sequences (even + * a null control sequence). + * + */ + while (i < n) { + if (c[i] == '{') { + braceLevel++; + if ((braceLevel == 1) && ((i + 1) < n) && (c[i + 1] == '\\')) { + i++; // skip brace + while ((i < n) && (braceLevel > 0)) { + i++; // skip backslash + + int afterBackslash = i; + while ((i < n) && Character.isLetter(c[i])) { + i++; + } + if ((i < n) && (i == afterBackslash)) { + i++; // Skip non-alpha control seq + } else { + if (BstCaseChanger.findSpecialChar(c, afterBackslash).isPresent()) { + result += BstWidthCalculator.getSpecialCharWidth(c, afterBackslash); + } + } + while ((i < n) && Character.isWhitespace(c[i])) { + i++; + } + while ((i < n) && (braceLevel > 0) && (c[i] != '\\')) { + if (c[i] == '}') { + braceLevel--; + } else if (c[i] == '{') { + braceLevel++; + } else { + result += BstWidthCalculator.getCharWidth(c[i]); + } + i++; + } + } + continue; + } + } else if (c[i] == '}') { + if (braceLevel > 0) { + braceLevel--; + } else { + LOGGER.warn("Too many closing braces in string: " + toMeasure); + } + } + result += BstWidthCalculator.getCharWidth(c[i]); + i++; + } + if (braceLevel > 0) { + LOGGER.warn("No enough closing braces in string: " + toMeasure); + } + return result; + } +} diff --git a/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java b/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java index c0f564800c0..1b98ae4aa15 100644 --- a/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java +++ b/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java @@ -110,7 +110,11 @@ protected void writeEntryTypeDefinition(BibEntryType customType) throws IOExcept @Override protected void writeProlog(BibDatabaseContext bibDatabaseContext, Charset encoding) throws IOException { - if ((encoding == null) || (encoding == StandardCharsets.UTF_8)) { + // We write the encoding if + // - it is provided (!= null) + // - explicitly set in the .bib file OR not equal to UTF_8 + // Otherwise, we do not write anything and return + if ((encoding == null) || (!bibDatabaseContext.getMetaData().getEncodingExplicitlySupplied() && (encoding == StandardCharsets.UTF_8))) { return; } diff --git a/src/main/java/org/jabref/logic/importer/ImportFormatReader.java b/src/main/java/org/jabref/logic/importer/ImportFormatReader.java index 617cd5c0dba..f3b8826171d 100644 --- a/src/main/java/org/jabref/logic/importer/ImportFormatReader.java +++ b/src/main/java/org/jabref/logic/importer/ImportFormatReader.java @@ -195,7 +195,7 @@ public UnknownFormatImport importUnknownFormat(Path filePath, FileUpdateMonitor parserResult.setPath(filePath); return new UnknownFormatImport(ImportFormatReader.BIBTEX_FORMAT, parserResult); } else { - throw new ImportException(Localization.lang("Could not find a suitable import format.")); + throw new ImportException(parserResult.getErrorMessage()); } } catch (IOException ignore) { // Ignored diff --git a/src/main/java/org/jabref/logic/importer/Importer.java b/src/main/java/org/jabref/logic/importer/Importer.java index f6ffc966214..56bde6f0e3e 100644 --- a/src/main/java/org/jabref/logic/importer/Importer.java +++ b/src/main/java/org/jabref/logic/importer/Importer.java @@ -123,12 +123,10 @@ protected static Charset getCharset(BufferedInputStream bufferedInputStream) { if ((matches == null) || (matches.length == 0)) { return defaultCharSet; } - if (Arrays.stream(matches).anyMatch(singleCharset -> singleCharset.getName().equals(defaultCharSet.toString()))) { + + if (Arrays.stream(matches).anyMatch(charset -> "ASCII".equals(charset.getName()))) { return defaultCharSet; } - if (Arrays.stream(matches).anyMatch(singleCharset -> singleCharset.getName().equals(StandardCharsets.UTF_16.toString()))) { - return StandardCharsets.UTF_16; - } if (matches[0] != null) { return Charset.forName(matches[0].getName()); @@ -166,7 +164,7 @@ public static BufferedReader getReader(Path filePath) throws IOException { return new BufferedReader(new InputStreamReader(stream, StandardCharsets.UTF_8)); } - public static BufferedReader getReader(InputStream stream) throws IOException { + public static BufferedReader getReader(InputStream stream) { BufferedInputStream bufferedInputStream = new BufferedInputStream(stream); Charset charset = getCharset(bufferedInputStream); InputStreamReader reader = new InputStreamReader(bufferedInputStream, charset); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java index 5fb3d8ed900..65301e2a7a1 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java @@ -184,7 +184,7 @@ private void parse(T entryType, Map fields) { } else if (isMethodToIgnore(method.getName())) { continue; } else if (method.getName().startsWith("get")) { - putIfValueNotNull(fields, FieldFactory.parseField(method.getName().replace("get", "")), (String) method.invoke(entryType)); + putIfValueNotNull(fields, FieldFactory.parseField(entryType, method.getName().replace("get", "")), (String) method.invoke(entryType)); } } catch (IllegalArgumentException | InvocationTargetException | IllegalAccessException e) { LOGGER.error("Could not invoke method", e); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java index e35fe6c73ad..590ae78de4e 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java @@ -4,7 +4,10 @@ import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; +import java.io.InputStreamReader; import java.nio.charset.Charset; +import java.nio.charset.CharsetDecoder; +import java.nio.charset.CodingErrorAction; import java.nio.file.Files; import java.nio.file.Path; import java.util.Objects; @@ -68,18 +71,28 @@ public ParserResult importDatabase(Path filePath) throws IOException { } Charset encoding; + boolean encodingExplicitlySupplied; try (BufferedReader reader = Files.newBufferedReader(filePath, detectedCharset)) { Optional suppliedEncoding = getSuppliedEncoding(reader); LOGGER.debug("Supplied encoding: {}", suppliedEncoding); + encodingExplicitlySupplied = suppliedEncoding.isPresent(); // in case no encoding information is present, use the detected one encoding = suppliedEncoding.orElse(detectedCharset); LOGGER.debug("Encoding used to read the file: {}", encoding); } - try (BufferedReader reader = Files.newBufferedReader(filePath, encoding)) { + // We replace unreadable characters + // Unfortunately, no warning will be issued to the user + // As this is a very seldom case, we accept that + CharsetDecoder decoder = encoding.newDecoder(); + decoder.onMalformedInput(CodingErrorAction.REPLACE); + + try (InputStreamReader inputStreamReader = new InputStreamReader(Files.newInputStream(filePath), decoder); + BufferedReader reader = new BufferedReader(inputStreamReader)) { ParserResult parserResult = this.importDatabase(reader); parserResult.getMetaData().setEncoding(encoding); + parserResult.getMetaData().setEncodingExplicitlySupplied(encodingExplicitlySupplied); parserResult.setPath(filePath); if (parserResult.getMetaData().getMode().isEmpty()) { parserResult.getMetaData().setMode(BibDatabaseModeDetection.inferMode(parserResult.getDatabase())); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java b/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java index 3fe4fb684d6..666ae01a7fb 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java @@ -283,7 +283,7 @@ private void parseAndAddEntry(String type) { LOGGER.debug("Could not parse entry", ex); parserResult.addWarning(Localization.lang("Error occurred when parsing entry") + ": '" + ex.getMessage() - + "'. " + Localization.lang("Skipped entry.")); + + "'. " + "\n\n" + Localization.lang("JabRef skipped the entry.")); } } @@ -387,8 +387,8 @@ private String purge(String context, String stringToPurge) { } // strip empty lines while ((runningIndex < indexOfAt) && - (context.charAt(runningIndex) == '\r' || - context.charAt(runningIndex) == '\n')) { + ((context.charAt(runningIndex) == '\r') || + (context.charAt(runningIndex) == '\n'))) { runningIndex++; } return context.substring(runningIndex); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java index 2ff451e6dca..afbe5af657d 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java @@ -14,6 +14,7 @@ import org.jabref.model.entry.Author; import org.jabref.model.entry.AuthorList; import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.BiblatexSoftwareField; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.StandardField; import org.jabref.model.entry.field.UnknownField; @@ -95,7 +96,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { StandardEntryType entryType = StandardEntryType.Software; // Map CFF fields to JabRef Fields - HashMap fieldMap = getFieldMappings(); + HashMap fieldMap = getFieldMappings(); for (Map.Entry property : citation.values.entrySet()) { if (fieldMap.containsKey(property.getKey())) { entryMap.put(fieldMap.get(property.getKey()), property.getValue()); @@ -120,7 +121,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { entryMap.put(StandardField.AUTHOR, authorStr); // Select DOI to keep - if (entryMap.get(StandardField.DOI) == null && citation.ids != null) { + if ((entryMap.get(StandardField.DOI) == null) && (citation.ids != null)) { List doiIds = citation.ids.stream() .filter(id -> id.type.equals("doi")) .collect(Collectors.toList()); @@ -137,14 +138,14 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { .collect(Collectors.toList()); if (swhIds.size() == 1) { - entryMap.put(StandardField.SWHID, swhIds.get(0)); + entryMap.put(BiblatexSoftwareField.SWHID, swhIds.get(0)); } else if (swhIds.size() > 1) { List relSwhIds = swhIds.stream() .filter(id -> id.split(":").length > 3) // quick filter for invalid swhids .filter(id -> id.split(":")[2].equals("rel")) .collect(Collectors.toList()); if (relSwhIds.size() == 1) { - entryMap.put(StandardField.SWHID, relSwhIds.get(0)); + entryMap.put(BiblatexSoftwareField.SWHID, relSwhIds.get(0)); } } } @@ -166,19 +167,19 @@ public boolean isRecognizedFormat(BufferedReader reader) throws IOException { try { citation = mapper.readValue(reader, CffFormat.class); - return citation != null && citation.values.get("title") != null; + return (citation != null) && (citation.values.get("title") != null); } catch (IOException e) { return false; } } - private HashMap getFieldMappings() { - HashMap fieldMappings = new HashMap<>(); + private HashMap getFieldMappings() { + HashMap fieldMappings = new HashMap<>(); fieldMappings.put("title", StandardField.TITLE); fieldMappings.put("version", StandardField.VERSION); fieldMappings.put("doi", StandardField.DOI); - fieldMappings.put("license", StandardField.LICENSE); - fieldMappings.put("repository", StandardField.REPOSITORY); + fieldMappings.put("license", BiblatexSoftwareField.LICENSE); + fieldMappings.put("repository", BiblatexSoftwareField.REPOSITORY); fieldMappings.put("url", StandardField.URL); fieldMappings.put("abstract", StandardField.ABSTRACT); fieldMappings.put("message", StandardField.COMMENT); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java index ab857becd0a..ab638ae4cf3 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java @@ -137,7 +137,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { } else if ("DT- ".equals(code)) { setOrAppend(b, new UnknownField("documenttype"), line.substring(4).trim(), ", "); } else { - setOrAppend(b, FieldFactory.parseField(code.substring(0, 2)), line.substring(4).trim(), ", "); + setOrAppend(b, FieldFactory.parseField(StandardEntryType.Book, line.substring(0, 2)), line.substring(4).trim(), ", "); } } results.add(b); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java index b502977f716..2845756b770 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java @@ -281,7 +281,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { if ("ER".equals(beg) || "EF".equals(beg) || "VR".equals(beg) || "FN".equals(beg)) { continue; } - hm.put(FieldFactory.parseField(beg), value); + hm.put(FieldFactory.parseField(type, beg), value); } } diff --git a/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java index 392fad9772b..b9e9eac39d4 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java @@ -425,7 +425,7 @@ private void addArticleIdList(Map fields, ArticleIdList articleId if ("pubmed".equals(id.getIdType())) { fields.put(StandardField.PMID, id.getContent()); } else { - fields.put(FieldFactory.parseField(id.getIdType()), id.getContent()); + fields.put(FieldFactory.parseField(StandardEntryType.Article, id.getIdType()), id.getContent()); } } } @@ -499,7 +499,7 @@ private void addKeyWords(Map fields, List allKeyword private void addOtherId(Map fields, List otherID) { for (OtherID id : otherID) { if ((id.getSource() != null) && (id.getContent() != null)) { - fields.put(FieldFactory.parseField(id.getSource()), id.getContent()); + fields.put(FieldFactory.parseField(StandardEntryType.Article, id.getSource()), id.getContent()); } } } diff --git a/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java b/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java index eb1e905adca..5f3d5e29c52 100644 --- a/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java +++ b/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java @@ -65,6 +65,7 @@ public MetaData parse(MetaData metaData, Map data, Character key String user = entry.getKey().substring(MetaData.FILE_DIRECTORY.length() + 1); metaData.setUserFileDirectory(user, getSingleItem(value)); } else if (entry.getKey().startsWith(MetaData.SELECTOR_META_PREFIX)) { + // edge case, it might be one special field e.g. article from biblatex-apa, but we can't distinguish this from any other field and rather prefer to handle it as UnknownField metaData.addContentSelector(ContentSelectors.parse(FieldFactory.parseField(entry.getKey().substring(MetaData.SELECTOR_META_PREFIX.length())), StringUtil.unquote(entry.getValue(), MetaData.ESCAPE_CHARACTER))); } else if (entry.getKey().startsWith(MetaData.FILE_DIRECTORY + "Latex-")) { // The user name comes directly after "FILE_DIRECTORYLatex-" diff --git a/src/main/java/org/jabref/logic/layout/format/NameFormatter.java b/src/main/java/org/jabref/logic/layout/format/NameFormatter.java index 919ff4df2d5..e8812235435 100644 --- a/src/main/java/org/jabref/logic/layout/format/NameFormatter.java +++ b/src/main/java/org/jabref/logic/layout/format/NameFormatter.java @@ -5,7 +5,7 @@ import java.util.Map; import java.util.Objects; -import org.jabref.logic.bst.BibtexNameFormatter; +import org.jabref.logic.bst.util.BstNameFormatter; import org.jabref.logic.layout.LayoutFormatter; import org.jabref.model.entry.AuthorList; @@ -86,7 +86,7 @@ private static String format(String toFormat, AuthorList al, String[] formats) { for (int i = 1; i <= al.getNumberOfAuthors(); i++) { for (int j = 1; j < formats.length; j += 2) { if ("*".equals(formats[j])) { - sb.append(BibtexNameFormatter.formatName(toFormat, i, formats[j + 1], null)); + sb.append(BstNameFormatter.formatName(toFormat, i, formats[j + 1])); break; } else { String[] range = formats[j].split("\\.\\."); @@ -112,7 +112,7 @@ private static String format(String toFormat, AuthorList al, String[] formats) { } if ((s <= i) && (i <= e)) { - sb.append(BibtexNameFormatter.formatName(toFormat, i, formats[j + 1], null)); + sb.append(BstNameFormatter.formatName(toFormat, i, formats[j + 1])); break; } } diff --git a/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java b/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java index 24e897ae926..8ebc6bddd04 100644 --- a/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java +++ b/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java @@ -2,6 +2,7 @@ import java.util.List; import java.util.Queue; +import java.util.Set; import java.util.concurrent.ConcurrentLinkedQueue; import org.jabref.gui.util.BackgroundTask; @@ -88,12 +89,17 @@ public void createIndex(PdfIndexer indexer) { enqueueTask(() -> indexer.createIndex()); } - public void addToIndex(PdfIndexer indexer, BibDatabaseContext databaseContext) { + public void updateIndex(PdfIndexer indexer, BibDatabaseContext databaseContext) { + Set pathsToRemove = indexer.getListOfFilePaths(); for (BibEntry entry : databaseContext.getEntries()) { for (LinkedFile file : entry.getFiles()) { enqueueTask(() -> indexer.addToIndex(entry, file, databaseContext)); + pathsToRemove.remove(file.getLink()); } } + for (String pathToRemove : pathsToRemove) { + enqueueTask(() -> indexer.removeFromIndex(pathToRemove)); + } } public void addToIndex(PdfIndexer indexer, BibEntry entry, BibDatabaseContext databaseContext) { @@ -108,7 +114,7 @@ public void addToIndex(PdfIndexer indexer, BibEntry entry, List link public void removeFromIndex(PdfIndexer indexer, BibEntry entry, List linkedFiles) { for (LinkedFile file : linkedFiles) { - enqueueTask(() -> indexer.removeFromIndex(entry, file)); + enqueueTask(() -> indexer.removeFromIndex(file.getLink())); } } diff --git a/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java b/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java index 3b91be44f71..975cf83b1ec 100644 --- a/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java +++ b/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java @@ -4,8 +4,10 @@ import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.attribute.BasicFileAttributes; +import java.util.HashSet; import java.util.List; import java.util.Optional; +import java.util.Set; import java.util.concurrent.TimeUnit; import org.jabref.gui.LibraryTab; @@ -25,6 +27,8 @@ import org.apache.lucene.index.IndexWriterConfig; import org.apache.lucene.index.Term; import org.apache.lucene.search.IndexSearcher; +import org.apache.lucene.search.MatchAllDocsQuery; +import org.apache.lucene.search.ScoreDoc; import org.apache.lucene.search.TermQuery; import org.apache.lucene.search.TopDocs; import org.apache.lucene.store.Directory; @@ -110,19 +114,16 @@ public void addToIndex(BibEntry entry, LinkedFile linkedFile, BibDatabaseContext } /** - * Removes a pdf file linked to one entry in the database from the index + * Removes a pdf file identified by its path from the index * - * @param entry the entry the file is linked to - * @param linkedFile the link to the file to be removed + * @param linkedFilePath the path to the file to be removed */ - public void removeFromIndex(BibEntry entry, LinkedFile linkedFile) { + public void removeFromIndex(String linkedFilePath) { try (IndexWriter indexWriter = new IndexWriter( directoryToIndex, new IndexWriterConfig( new EnglishStemAnalyzer()).setOpenMode(IndexWriterConfig.OpenMode.CREATE_OR_APPEND))) { - if (!entry.getFiles().isEmpty()) { - indexWriter.deleteDocuments(new Term(SearchFieldConstants.PATH, linkedFile.getLink())); - } + indexWriter.deleteDocuments(new Term(SearchFieldConstants.PATH, linkedFilePath)); indexWriter.commit(); } catch (IOException e) { LOGGER.warn("Could not initialize the IndexWriter!", e); @@ -145,7 +146,7 @@ public void removeFromIndex(BibEntry entry) { */ public void removeFromIndex(BibEntry entry, List linkedFiles) { for (LinkedFile linkedFile : linkedFiles) { - removeFromIndex(entry, linkedFile); + removeFromIndex(linkedFile.getLink()); } } @@ -224,4 +225,25 @@ private void writeToIndex(BibEntry entry, LinkedFile linkedFile) { LOGGER.warn("Could not add the document {} to the index!", linkedFile.getLink(), e); } } + + /** + * Lists the paths of all the files that are stored in the index + * + * @return all file paths + */ + public Set getListOfFilePaths() { + Set paths = new HashSet<>(); + try (IndexReader reader = DirectoryReader.open(directoryToIndex)) { + IndexSearcher searcher = new IndexSearcher(reader); + MatchAllDocsQuery query = new MatchAllDocsQuery(); + TopDocs allDocs = searcher.search(query, Integer.MAX_VALUE); + for (ScoreDoc scoreDoc : allDocs.scoreDocs) { + Document doc = reader.document(scoreDoc.doc); + paths.add(doc.getField(SearchFieldConstants.PATH).stringValue()); + } + } catch (IOException e) { + return paths; + } + return paths; + } } diff --git a/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java b/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java index 260ec297c24..4b1a14fe1c9 100644 --- a/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java +++ b/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java @@ -25,6 +25,8 @@ public class SharedDatabasePreferences { private static final String SHARED_DATABASE_NAME = "sharedDatabaseName"; private static final String SHARED_DATABASE_USER = "sharedDatabaseUser"; private static final String SHARED_DATABASE_PASSWORD = "sharedDatabasePassword"; + private static final String SHARED_DATABASE_FOLDER = "sharedDatabaseFolder"; + private static final String SHARED_DATABASE_AUTOSAVE = "sharedDatabaseAutosave"; private static final String SHARED_DATABASE_REMEMBER_PASSWORD = "sharedDatabaseRememberPassword"; private static final String SHARED_DATABASE_USE_SSL = "sharedDatabaseUseSSL"; private static final String SHARED_DATABASE_KEYSTORE_FILE = "sharedDatabaseKeyStoreFile"; @@ -77,6 +79,14 @@ public boolean getRememberPassword() { return internalPrefs.getBoolean(SHARED_DATABASE_REMEMBER_PASSWORD, false); } + public Optional getFolder() { + return getOptionalValue(SHARED_DATABASE_FOLDER); + } + + public boolean getAutosave() { + return internalPrefs.getBoolean(SHARED_DATABASE_AUTOSAVE, false); + } + public boolean isUseSSL() { return internalPrefs.getBoolean(SHARED_DATABASE_USE_SSL, false); } @@ -109,6 +119,14 @@ public void setRememberPassword(boolean rememberPassword) { internalPrefs.putBoolean(SHARED_DATABASE_REMEMBER_PASSWORD, rememberPassword); } + public void setFolder(String folder) { + internalPrefs.put(SHARED_DATABASE_FOLDER, folder); + } + + public void setAutosave(boolean autosave) { + internalPrefs.putBoolean(SHARED_DATABASE_AUTOSAVE, autosave); + } + public void setUseSSL(boolean useSSL) { internalPrefs.putBoolean(SHARED_DATABASE_USE_SSL, useSSL); } diff --git a/src/main/java/org/jabref/model/database/BibDatabase.java b/src/main/java/org/jabref/model/database/BibDatabase.java index 72a559ff663..52e4e841ce9 100644 --- a/src/main/java/org/jabref/model/database/BibDatabase.java +++ b/src/main/java/org/jabref/model/database/BibDatabase.java @@ -20,6 +20,7 @@ import javafx.collections.FXCollections; import javafx.collections.ObservableList; +import javafx.collections.ObservableSet; import org.jabref.logic.bibtex.FieldWriter; import org.jabref.model.database.event.EntriesAddedEvent; @@ -29,6 +30,7 @@ import org.jabref.model.entry.Month; import org.jabref.model.entry.event.EntriesEventSource; import org.jabref.model.entry.event.EntryChangedEvent; +import org.jabref.model.entry.event.FieldAddedOrRemovedEvent; import org.jabref.model.entry.event.FieldChangedEvent; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.FieldFactory; @@ -52,6 +54,8 @@ public class BibDatabase { * State attributes */ private final ObservableList entries = FXCollections.synchronizedObservableList(FXCollections.observableArrayList(BibEntry::getObservables)); + + private final ObservableSet visibleFields = FXCollections.observableSet(); private Map bibtexStrings = new ConcurrentHashMap<>(); private final EventBus eventBus = new EventBus(); @@ -136,13 +140,8 @@ public ObservableList getEntries() { * * @return set of fieldnames, that are visible */ - public Set getAllVisibleFields() { - Set allFields = new TreeSet<>(Comparator.comparing(Field::getName)); - for (BibEntry e : getEntries()) { - allFields.addAll(e.getFields()); - } - return allFields.stream().filter(field -> !FieldFactory.isInternalField(field)) - .collect(Collectors.toSet()); + public ObservableSet getAllVisibleFields() { + return visibleFields; } /** @@ -214,6 +213,8 @@ public synchronized void insertEntries(List newEntries, EntriesEventSo eventBus.post(new EntriesAddedEvent(newEntries, newEntries.get(0), eventSource)); } entries.addAll(newEntries); + + updateVisibleFields(); } public synchronized void removeEntry(BibEntry bibEntry) { @@ -251,6 +252,7 @@ public synchronized void removeEntries(List toBeDeleted, EntriesEventS boolean anyRemoved = entries.removeIf(entry -> ids.contains(entry.getId())); if (anyRemoved) { eventBus.post(new EntriesRemovedEvent(toBeDeleted, eventSource)); + updateVisibleFields(); } } @@ -584,6 +586,30 @@ private void relayEntryChangeEvent(FieldChangedEvent event) { eventBus.post(event); } + @Subscribe + private void listen(FieldAddedOrRemovedEvent event) { + // When a field is removed from an entry we can't tell if it's + // still present in other entries, and thus we can't remove it + // from the set of visible fields. However, when a new field is added + // to any entry, we can simply add it to the set because we're + // going to add it whether other entries have it or not + boolean isAdded = visibleFields.add(event.getField()); + if (!isAdded) { + updateVisibleFields(); + } + } + + private void updateVisibleFields() { + visibleFields.clear(); + Set allFields = new TreeSet<>(Comparator.comparing(Field::getName)); + for (BibEntry e : getEntries()) { + allFields.addAll(e.getFields()); + } + visibleFields.addAll(allFields.stream().filter(field -> !FieldFactory.isInternalField(field)) + .filter(field -> StringUtil.isNotBlank(field.getName())) + .collect(Collectors.toSet())); + } + public Optional getReferencedEntry(BibEntry entry) { return entry.getField(StandardField.CROSSREF).flatMap(this::getEntryByCitationKey); } diff --git a/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java b/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java index b737596abd6..836cd39dac8 100644 --- a/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java +++ b/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java @@ -11,6 +11,7 @@ import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.field.BibField; import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.types.BiblatexAPAEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexSoftwareEntryTypeDefinitions; import org.jabref.model.entry.types.BibtexEntryTypeDefinitions; @@ -21,7 +22,7 @@ public class BibEntryTypesManager { public static final String ENTRYTYPE_FLAG = "jabref-entrytype: "; private final InternalEntryTypes BIBTEX = new InternalEntryTypes(Stream.concat(BibtexEntryTypeDefinitions.ALL.stream(), IEEETranEntryTypeDefinitions.ALL.stream()).collect(Collectors.toList())); - private final InternalEntryTypes BIBLATEX = new InternalEntryTypes(Stream.concat(BiblatexEntryTypeDefinitions.ALL.stream(), BiblatexSoftwareEntryTypeDefinitions.ALL.stream()).collect(Collectors.toList())); + private final InternalEntryTypes BIBLATEX = new InternalEntryTypes(Stream.concat(BiblatexEntryTypeDefinitions.ALL.stream(), Stream.concat(BiblatexSoftwareEntryTypeDefinitions.ALL.stream(), BiblatexAPAEntryTypeDefinitions.ALL.stream())).collect(Collectors.toList())); public BibEntryTypesManager() { } @@ -99,6 +100,7 @@ public List getAllCustomTypes(BibDatabaseMode mode) { return customizedTypes.stream() .filter(entryType -> BiblatexEntryTypeDefinitions.ALL.stream().noneMatch(biblatexType -> biblatexType.getType().equals(entryType.getType()))) .filter(entryType -> BiblatexSoftwareEntryTypeDefinitions.ALL.stream().noneMatch(biblatexSoftware -> biblatexSoftware.getType().equals(entryType.getType()))) + .filter(entryType -> BiblatexAPAEntryTypeDefinitions.ALL.stream().noneMatch(biblatexAPA -> biblatexAPA.getType().equals(entryType.getType()))) .collect(Collectors.toList()); } } diff --git a/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java b/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java new file mode 100644 index 00000000000..b2938963efe --- /dev/null +++ b/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java @@ -0,0 +1,82 @@ +package org.jabref.model.entry.field; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; +import java.util.Optional; +import java.util.Set; + +import org.jabref.model.entry.types.BiblatexApaEntryType; + +public enum BiblatexApaField implements Field { + + AMENDMENT("amendment"), + ARTICLE("article"), + CITATION("citation"), + CITATION_CITEORG("citation_citeorg"), + CITATION_CITEDATE("citation_citedate", FieldProperty.DATE), + CITATION_CITEINFO("citation_citeinfo"), + SECTION("section", FieldProperty.NUMERIC), + SOURCE("source"); + + private final String name; + private final String displayName; + private final Set properties; + + BiblatexApaField(String name) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexApaField(String name, String displayName) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexApaField(String name, String displayName, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.of(first, rest); + } + + BiblatexApaField(String name, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.of(first, rest); + } + + public static Optional fromName(T type, String name) { + if (!(type instanceof BiblatexApaEntryType)) { + return Optional.empty(); + } + return Arrays.stream(BiblatexApaField.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public Set getProperties() { + return Collections.unmodifiableSet(properties); + } + + @Override + public String getName() { + return name; + } + + @Override + public boolean isStandardField() { + return false; + } + + @Override + public String getDisplayName() { + if (displayName == null) { + return Field.super.getDisplayName(); + } else { + return displayName; + } + } +} diff --git a/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java b/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java new file mode 100644 index 00000000000..fc3bb4dcc48 --- /dev/null +++ b/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java @@ -0,0 +1,82 @@ +package org.jabref.model.entry.field; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; +import java.util.Optional; +import java.util.Set; + +import org.jabref.model.entry.types.BiblatexSoftwareEntryType; + +public enum BiblatexSoftwareField implements Field { + + HALID("hal_id"), + HALVERSION("hal_version"), + INTRODUCEDIN("introducedin"), + LICENSE("license"), + RELATEDTYPE("relatedtype"), + RELATEDSTRING("relatedstring"), + REPOSITORY("repository"), + SWHID("swhid"); + + private final String name; + private final String displayName; + private final Set properties; + + BiblatexSoftwareField(String name) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexSoftwareField(String name, String displayName) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexSoftwareField(String name, String displayName, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.of(first, rest); + } + + BiblatexSoftwareField(String name, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.of(first, rest); + } + + public static Optional fromName(T type, String name) { + if (!(type instanceof BiblatexSoftwareEntryType)) { + return Optional.empty(); + } + return Arrays.stream(BiblatexSoftwareField.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public Set getProperties() { + return Collections.unmodifiableSet(properties); + } + + @Override + public String getName() { + return name; + } + + @Override + public boolean isStandardField() { + return false; + } + + @Override + public String getDisplayName() { + if (displayName == null) { + return Field.super.getDisplayName(); + } else { + return displayName; + } + } +} diff --git a/src/main/java/org/jabref/model/entry/field/FieldFactory.java b/src/main/java/org/jabref/model/entry/field/FieldFactory.java index b1c2f3b99b9..07a34b039d3 100644 --- a/src/main/java/org/jabref/model/entry/field/FieldFactory.java +++ b/src/main/java/org/jabref/model/entry/field/FieldFactory.java @@ -73,13 +73,23 @@ public static String serializeFieldsList(Collection fields) { .collect(Collectors.joining(DELIMITER)); } + public static Field parseField(T type, String fieldName) { + return OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + InternalField.fromName(fieldName), + StandardField.fromName(fieldName)), + SpecialField.fromName(fieldName)), + IEEEField.fromName(fieldName)), + BiblatexSoftwareField.fromName(type, fieldName)), + BiblatexApaField.fromName(type, fieldName)) + .orElse(new UnknownField(fieldName)); + } + public static Field parseField(String fieldName) { - return OptionalUtil.orElse(OptionalUtil.orElse(OptionalUtil.orElse( - InternalField.fromName(fieldName), - StandardField.fromName(fieldName)), - SpecialField.fromName(fieldName)), - IEEEField.fromName(fieldName)) - .orElse(new UnknownField(fieldName)); + return parseField(null, fieldName); } public static Set getKeyFields() { @@ -138,6 +148,8 @@ private static Set getFieldsFiltered(Predicate selector) { private static Set getAllFields() { Set fields = new HashSet<>(); + fields.addAll(EnumSet.allOf(BiblatexApaField.class)); + fields.addAll(EnumSet.allOf(BiblatexSoftwareField.class)); fields.addAll(EnumSet.allOf(IEEEField.class)); fields.addAll(EnumSet.allOf(InternalField.class)); fields.addAll(EnumSet.allOf(SpecialField.class)); diff --git a/src/main/java/org/jabref/model/entry/field/IEEEField.java b/src/main/java/org/jabref/model/entry/field/IEEEField.java index d00bec02eee..f56ce79ea99 100644 --- a/src/main/java/org/jabref/model/entry/field/IEEEField.java +++ b/src/main/java/org/jabref/model/entry/field/IEEEField.java @@ -31,7 +31,7 @@ public enum IEEEField implements Field { this.properties = EnumSet.of(first, rest); } - public static Optional fromName(String name) { + public static Optional fromName(String name) { return Arrays.stream(IEEEField.values()) .filter(field -> field.getName().equalsIgnoreCase(name)) .findAny(); diff --git a/src/main/java/org/jabref/model/entry/field/StandardField.java b/src/main/java/org/jabref/model/entry/field/StandardField.java index ef3646802e0..bcf9b710b7d 100644 --- a/src/main/java/org/jabref/model/entry/field/StandardField.java +++ b/src/main/java/org/jabref/model/entry/field/StandardField.java @@ -59,14 +59,11 @@ public enum StandardField implements Field { FOREWORD("foreword", FieldProperty.PERSON_NAMES), FOLDER("folder"), GENDER("gender", FieldProperty.GENDER), - HALID("hal_id"), - HALVERSION("hal_version"), HOLDER("holder", FieldProperty.PERSON_NAMES), HOWPUBLISHED("howpublished"), IDS("ids", FieldProperty.MULTIPLE_ENTRY_LINK), INSTITUTION("institution"), INTRODUCTION("introduction", FieldProperty.PERSON_NAMES), - INTRODUCEDIN("introducedin"), ISBN("isbn", "ISBN", FieldProperty.ISBN), ISRN("isrn", "ISRN"), ISSN("issn", "ISSN"), @@ -81,7 +78,6 @@ public enum StandardField implements Field { LANGUAGE("language", FieldProperty.LANGUAGE), LABEL("label"), LIBRARY("library"), - LICENSE("license"), LOCATION("location"), MAINSUBTITLE("mainsubtitle", FieldProperty.BOOK_NAME), MAINTITLE("maintitle", FieldProperty.BOOK_NAME), @@ -106,10 +102,7 @@ public enum StandardField implements Field { PUBSTATE("pubstate", FieldProperty.PUBLICATION_STATE), PRIMARYCLASS("primaryclass"), RELATED("related", FieldProperty.MULTIPLE_ENTRY_LINK), - RELATEDTYPE("relatedtype"), - RELATEDSTRING("relatedstring"), REPORTNO("reportno"), - REPOSITORY("repository"), REVIEW("review"), REVISION("revision"), SCHOOL("school"), @@ -120,7 +113,6 @@ public enum StandardField implements Field { SORTKEY("sortkey"), SORTNAME("sortname", FieldProperty.PERSON_NAMES), SUBTITLE("subtitle"), - SWHID("swhid"), TITLE("title"), TITLEADDON("titleaddon"), TRANSLATOR("translator", FieldProperty.PERSON_NAMES), diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java b/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java new file mode 100644 index 00000000000..f5c0b89b0be --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java @@ -0,0 +1,43 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.List; + +import org.jabref.model.entry.BibEntryType; +import org.jabref.model.entry.BibEntryTypeBuilder; +import org.jabref.model.entry.field.BiblatexApaField; +import org.jabref.model.entry.field.StandardField; + +public class BiblatexAPAEntryTypeDefinitions { + + private static final BibEntryType JURISDICTION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Jurisdiction) + .withImportantFields(StandardField.ORGANIZATION, BiblatexApaField.CITATION_CITEORG, BiblatexApaField.CITATION_CITEDATE, BiblatexApaField.CITATION_CITEDATE, StandardField.ORIGDATE) + .withRequiredFields(StandardField.TITLE, BiblatexApaField.CITATION, BiblatexApaField.CITATION_CITEINFO, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType LEGISLATION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legislation) + .withImportantFields(StandardField.TITLEADDON, StandardField.ORIGDATE) + .withRequiredFields(StandardField.TITLE, StandardField.LOCATION, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType LEGADMINMATERIAL = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legadminmaterial) + .withImportantFields(StandardField.NUMBER, StandardField.SHORTTITLE, StandardField.NOTE, StandardField.KEYWORDS) + .withRequiredFields(StandardField.TITLE, BiblatexApaField.CITATION, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType CONSTITUTION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Constitution) + .withImportantFields(BiblatexApaField.ARTICLE, BiblatexApaField.AMENDMENT, StandardField.EVENTDATE, StandardField.KEYWORDS, StandardField.PART, BiblatexApaField.SECTION) + .withRequiredFields(BiblatexApaField.SOURCE, StandardField.TYPE) + .build(); + + private static final BibEntryType LEGAL = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legal) + .withRequiredFields(StandardField.TITLE, StandardField.DATE, StandardField.URI, StandardField.KEYWORDS, StandardField.PART, BiblatexApaField.SECTION) + .build(); + + public static final List ALL = Arrays.asList(JURISDICTION, LEGISLATION, LEGADMINMATERIAL, CONSTITUTION, LEGAL); +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java b/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java new file mode 100644 index 00000000000..6a1d8d3abe2 --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java @@ -0,0 +1,36 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.Locale; +import java.util.Optional; + +public enum BiblatexApaEntryType implements EntryType { + + Legislation("Legislation"), + Legadminmaterial("Legadminmaterial"), + Jurisdiction("Jurisdiction"), + Constitution("Constitution"), + Legal("Legal"); + + private final String displayName; + + BiblatexApaEntryType(String displayName) { + this.displayName = displayName; + } + + @Override + public String getName() { + return displayName.toLowerCase(Locale.ENGLISH); + } + + @Override + public String getDisplayName() { + return displayName; + } + + public static Optional fromName(String name) { + return Arrays.stream(BiblatexApaEntryType.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java new file mode 100644 index 00000000000..041c48e592a --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java @@ -0,0 +1,35 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.Locale; +import java.util.Optional; + +public enum BiblatexSoftwareEntryType implements EntryType { + + Dataset("Dataset"), + SoftwareVersion("SoftwareVersion"), + SoftwareModule("SoftwareModule"), + CodeFragment("CodeFragment"); + + private final String displayName; + + BiblatexSoftwareEntryType(String displayName) { + this.displayName = displayName; + } + + public static Optional fromName(String name) { + return Arrays.stream(BiblatexSoftwareEntryType.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public String getName() { + return displayName.toLowerCase(Locale.ENGLISH); + } + + @Override + public String getDisplayName() { + return displayName; + } +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java index 0fc48e6d904..6a2e7f0851e 100644 --- a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java +++ b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java @@ -5,6 +5,7 @@ import org.jabref.model.entry.BibEntryType; import org.jabref.model.entry.BibEntryTypeBuilder; +import org.jabref.model.entry.field.BiblatexSoftwareField; import org.jabref.model.entry.field.OrFields; import org.jabref.model.entry.field.StandardField; @@ -12,40 +13,40 @@ public class BiblatexSoftwareEntryTypeDefinitions { private static final BibEntryType SOFTWARE = new BibEntryTypeBuilder() .withType(StandardEntryType.Software) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.VERSION, StandardField.YEAR) .build(); private static final BibEntryType SOFTWAREVERSION = new BibEntryTypeBuilder() - .withType(StandardEntryType.SoftwareVersion) - .withImportantFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, StandardField.HALID, StandardField.HALVERSION, - StandardField.INSTITUTION, StandardField.INTRODUCEDIN, StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, - StandardField.PUBLISHER, StandardField.RELATED, StandardField.RELATEDTYPE, StandardField.RELATEDSTRING, - StandardField.REPOSITORY, StandardField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) + .withType(BiblatexSoftwareEntryType.SoftwareVersion) + .withImportantFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, + StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, + StandardField.PUBLISHER, StandardField.RELATED, BiblatexSoftwareField.RELATEDTYPE, BiblatexSoftwareField.RELATEDSTRING, + BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.YEAR, StandardField.VERSION) - .withDetailFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, StandardField.HALID, StandardField.HALVERSION, - StandardField.INSTITUTION, StandardField.INTRODUCEDIN, StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, - StandardField.PUBLISHER, StandardField.RELATED, StandardField.RELATEDTYPE, StandardField.RELATEDSTRING, - StandardField.REPOSITORY, StandardField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) + .withDetailFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, + StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, + StandardField.PUBLISHER, StandardField.RELATED, BiblatexSoftwareField.RELATEDTYPE, BiblatexSoftwareField.RELATEDSTRING, + BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.YEAR) .build(); private static final BibEntryType SOFTWAREMODULE = new BibEntryTypeBuilder() - .withType(StandardEntryType.SoftwareModule) + .withType(BiblatexSoftwareEntryType.SoftwareModule) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(StandardField.AUTHOR, StandardField.SUBTITLE, StandardField.URL, StandardField.YEAR) .build(); private static final BibEntryType CODEFRAGMENT = new BibEntryTypeBuilder() - .withType(StandardEntryType.CodeFragment) + .withType(BiblatexSoftwareEntryType.CodeFragment) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(StandardField.URL) .build(); diff --git a/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java b/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java index 1c94f6dd9a0..6b40a56c4f4 100644 --- a/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java +++ b/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java @@ -49,6 +49,8 @@ private static boolean isBiblatex(EntryType type) { public static EntryType parse(String typeName) { List types = new ArrayList<>(Arrays.asList(StandardEntryType.values())); types.addAll(Arrays.asList(IEEETranEntryType.values())); + types.addAll(Arrays.asList(BiblatexSoftwareEntryType.values())); + types.addAll(Arrays.asList(BiblatexApaEntryType.values())); types.addAll(Arrays.asList(SystematicLiteratureReviewStudyEntryType.values())); return types.stream().filter(type -> type.getName().equals(typeName.toLowerCase(Locale.ENGLISH))).findFirst().orElse(new UnknownEntryType(typeName)); diff --git a/src/main/java/org/jabref/model/entry/types/StandardEntryType.java b/src/main/java/org/jabref/model/entry/types/StandardEntryType.java index fb2922e61a0..6f6f91ac6a5 100644 --- a/src/main/java/org/jabref/model/entry/types/StandardEntryType.java +++ b/src/main/java/org/jabref/model/entry/types/StandardEntryType.java @@ -36,10 +36,7 @@ public enum StandardEntryType implements EntryType { Thesis("Thesis"), WWW("WWW"), Software("Software"), - Dataset("Dataset"), - SoftwareVersion("SoftwareVersion"), - SoftwareModule("SoftwareModule"), - CodeFragment("CodeFragment"); + Dataset("Dataset"); private final String displayName; diff --git a/src/main/java/org/jabref/model/metadata/MetaData.java b/src/main/java/org/jabref/model/metadata/MetaData.java index 238658803d3..9b0ed2ae856 100644 --- a/src/main/java/org/jabref/model/metadata/MetaData.java +++ b/src/main/java/org/jabref/model/metadata/MetaData.java @@ -68,6 +68,7 @@ public class MetaData { private final ContentSelectors contentSelectors = new ContentSelectors(); private final Map> unknownMetaData = new HashMap<>(); private boolean isEventPropagationEnabled = true; + private boolean encodingExplicitlySupplied; /** * Constructs an empty metadata. @@ -291,6 +292,17 @@ public void setEncoding(Charset encoding, ChangePropagation postChanges) { } } + public boolean getEncodingExplicitlySupplied() { + return encodingExplicitlySupplied; + } + + /** + * Sets the indication whether the encoding was set using "% Encoding: ..." or whether it was detected "magically" + */ + public void setEncodingExplicitlySupplied(boolean encodingExplicitlySupplied) { + this.encodingExplicitlySupplied = encodingExplicitlySupplied; + } + /** * If disabled {@link MetaDataChangedEvent} will not be posted. */ @@ -349,6 +361,7 @@ public boolean equals(Object o) { return (isProtected == metaData.isProtected) && Objects.equals(groupsRoot.getValue(), metaData.groupsRoot.getValue()) && Objects.equals(encoding, metaData.encoding) + && Objects.equals(encodingExplicitlySupplied, metaData.encodingExplicitlySupplied) && Objects.equals(saveOrderConfig, metaData.saveOrderConfig) && Objects.equals(citeKeyPatterns, metaData.citeKeyPatterns) && Objects.equals(userFileDirectory, metaData.userFileDirectory) @@ -362,7 +375,7 @@ public boolean equals(Object o) { @Override public int hashCode() { - return Objects.hash(groupsRoot.getValue(), encoding, saveOrderConfig, citeKeyPatterns, userFileDirectory, + return Objects.hash(groupsRoot.getValue(), encoding, encodingExplicitlySupplied, saveOrderConfig, citeKeyPatterns, userFileDirectory, defaultCiteKeyPattern, saveActions, mode, isProtected, defaultFileDirectory); } } diff --git a/src/main/java/org/jabref/model/strings/StringUtil.java b/src/main/java/org/jabref/model/strings/StringUtil.java index 19f0b7a4542..2a443306e4d 100644 --- a/src/main/java/org/jabref/model/strings/StringUtil.java +++ b/src/main/java/org/jabref/model/strings/StringUtil.java @@ -750,4 +750,15 @@ public static String quoteStringIfSpaceIsContained(String string) { return string; } } + + /** + * Checks if the given string contains any whitespace characters. The supported whitespace characters + * are the set of characters matched by {@code \s} in regular expressions, which are {@code [ \t\n\x0B\f\r]}. + * + * @param s The string to check + * @return {@code True} if the given string does contain at least one whitespace character, {@code False} otherwise + * */ + public static boolean containsWhitespace(String s) { + return s.chars().anyMatch(Character::isWhitespace); + } } diff --git a/src/main/resources/l10n/JabRef_da.properties b/src/main/resources/l10n/JabRef_da.properties index c07cb809a45..97eeb562207 100644 --- a/src/main/resources/l10n/JabRef_da.properties +++ b/src/main/resources/l10n/JabRef_da.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 matcher regulærudtrykket %1 %0\ matches\ the\ term\ %1=%0 matcher udtrykket %1 @@ -122,8 +123,6 @@ Cite\ command=Citations-kommando Clear=Ryd -Clear\ fields=Ryd felter - Close\ dialog=Luk dialog @@ -574,11 +573,18 @@ Remove\ entry\ from\ import=Fjern post fra import Remove\ group=Fjern gruppe -Remove\ group\ "%0"?=Fjern gruppen "%0"? +Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Fjern gruppen "%0" og dens undergrupper? -Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernede gruppen "%0" og dens undergrupper + + Remove\ link=Slet link @@ -587,8 +593,6 @@ Remove\ old\ entry=Fjern gammel post Removed\ group\ "%0".=Fjernede gruppen "%0" -Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernede gruppen "%0" og dens undergrupper - Removed\ string=Streng fjernet Renamed\ string=ændrede navn på streng @@ -642,9 +646,6 @@ Select\ file\ from\ ZIP-archive=Vælg fil fra ZIP-fil Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Vælg forgreningerne for at inspicere og acceptere eller forkaste ændringer -Set\ field=Sæt felt -Set\ fields=Sæt felter - Settings=Indstillinger Shortcut=Genvej @@ -675,7 +676,6 @@ Size=Størrelse Skipped\ -\ No\ PDF\ linked=Sprang over - ingen PDF-fil linket Skipped\ -\ PDF\ does\ not\ exist=Sprang over - PDF-filen findes ikke -Skipped\ entry.=Sprang over post. source\ edit=redigering af kilde @@ -787,8 +787,6 @@ MIME\ type=MIME-type This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ running\ instance\ of\ JabRef\ instead\ of\ opening\ a\ new\ instance.\ For\ instance,\ this\ is\ useful\ when\ you\ open\ a\ file\ in\ JabRef\ from\ your\ web\ browser.\ Note\ that\ this\ will\ prevent\ you\ from\ running\ more\ than\ one\ instance\ of\ JabRef\ at\ a\ time.=Denne funktion tillader, at flere filer kan åbnes eller importeres i en allerede kørende JabRef i stedet for at åbne programmet påny. For eksempel er dette praktisk, når du åbner filer i JabRef fra din web browser. Bemærk at dette vil forhindre dig i at køre mere end en instans af JabRef ad gangen. Rename\ field=Omdøb felt -Rename\ field\ to=Omdøb felt til -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flyt indhold af et felt til et felt med et andet navn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan ikke bruge port %0 til fjernstyring; et andet program bruger den måske. Prøv en anden port. @@ -914,6 +912,7 @@ Finished\ automatically\ setting\ external\ links.=Fuldførte automatisk udfyldn + Open\ folder=Åben mappe @@ -1088,6 +1087,8 @@ Default\ pattern=Standardmønster + + diff --git a/src/main/resources/l10n/JabRef_de.properties b/src/main/resources/l10n/JabRef_de.properties index d9f4e1272f6..ee962e6a73b 100644 --- a/src/main/resources/l10n/JabRef_de.properties +++ b/src/main/resources/l10n/JabRef_de.properties @@ -19,7 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 Einträge -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 Export erfolgreich. Ordner mit der gespeicherten Datei öffnen? +Export\ operation\ finished\ successfully.=Exportvorgang erfolgreich abgeschlossen. + +Reveal\ in\ File\ Explorer=Im Datei-Explorer anzeigen %0\ matches\ the\ regular\ expression\ %1=%0 exakt dem regulären Ausdruck %1 entspricht @@ -151,8 +153,6 @@ Cite\ command=Cite-Befehl Clear=Zurücksetzen -Clear\ fields=Felder löschen - Open\ /\ close\ entry\ editor=Eintragseditor öffnen / schließen Close\ dialog=Dialog schließen @@ -703,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Ausgewählte Einträge aus dieser G Remove\ group=Gruppe löschen -Remove\ group\ "%0"?=Gruppe "%0" löschen? +Remove\ group\ and\ subgroups=Gruppe und Untergruppen löschen + +Remove\ groups\ and\ subgroups=Gruppen und Untergruppen entfernen + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Alle ausgewählten Gruppen entfernen und ihre Untergruppen beibehalten? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Gruppe "%0" entfernen und Untergruppen beibehalten? + +Remove\ groups=Gruppen entfernen + +Removed\ all\ selected\ groups.=Alle ausgewählten Gruppen entfernen. Remove\ group\ "%0"\ and\ its\ subgroups?=Gruppe "%0" inklusive Untergruppen löschen? -Remove\ group\ and\ subgroups=Gruppe und Untergruppen löschen +Removed\ group\ "%0"\ and\ its\ subgroups.=Gruppe "%0" inklusive Untergruppen gelöscht. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Alle ausgewählten Gruppen und ihre Untergruppen entfernen? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Alle ausgewählten Gruppen und ihre Untergruppen entfernt. Remove\ link=Link löschen @@ -717,8 +731,6 @@ Remove\ string\ %0=Entferne String %0 Removed\ group\ "%0".=Gruppe "%0" gelöscht. -Removed\ group\ "%0"\ and\ its\ subgroups.=Gruppe "%0" inklusive Untergruppen gelöscht. - Removed\ string=String gelöscht Renamed\ string=String umbenannt @@ -790,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Eintrag aus der ZIP-Archiv auswählen Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Wählen Sie die Verzweigungen aus, um die Änderungen zu sehen und anzunehmen oder zu verwerfen -Set\ field=Setze Feld -Set\ fields=Felder setzen - Settings=Einstellungen Shortcut=Tastenkürzel @@ -825,7 +834,12 @@ Size=Größe Skipped\ -\ No\ PDF\ linked=Übersprungen - Kein PDF verlinkt Skipped\ -\ PDF\ does\ not\ exist=Übersprungen - PDF exisitert nicht -Skipped\ entry.=Eintrag übersprungen. +JabRef\ skipped\ the\ entry.=JabRef hat den Eintrag übersprungen. +Import\ error=Fehler beim Importieren +Open\ library\ error=Fehler beim Öffnen der Bibliothek +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Bitte überprüfen Sie Ihre Bibliotheksdatei auf falsche Syntax. +SourceTab\ error=SourceTab Fehler +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Benutzereingabe via Eintragseditor im Tab `{}bibtex source` führte zu einem Fehler. Sort\ subgroups=Untergruppen sortieren @@ -972,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Zeile %0\: Be No\ full\ text\ document\ found=Kein Volltext-Dokument gefunden Download\ from\ URL=Download von URL Rename\ field=Feld umbenennen -Append\ field=An Feld anfügen -Append\ to\ fields=An Felder anfügen -Rename\ field\ to=Feld umbenennen -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Inhalt eines Felds in ein Feld mit anderem Namen verschieben Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Port %0 konnte nicht für externen Zugriff genutzt werden; er wird möglicherweise von einer anderen Anwendung benutzt. Versuchen Sie einen anderen Port. @@ -1243,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Benutzerdefinierte Proxy-Konfiguration verwend Proxy\ requires\ authentication=Proxy erfordert Authentifizierung Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Achtung\: Das Passwort wird im Klartext gespeichert\! Clear\ connection\ settings=Verbindungseinstellungen zurücksetzen +Check\ Proxy\ Setting=Überprüfe Proxy-Einstellungen +Check\ connection=Verbindung prüfen +Connection\ failed\!=Verbindung fehlgeschlagen\! +Connection\ successful\!=Verbindung erfolgreich\! + +SSL\ Configuration=SSL-Konfiguration +SSL\ configuration\ changed=SSL-Konfiguration geändert +SSL\ certificate\ file=SSL-Zertifikatsdatei +Duplicate\ Certificates=Doppelte Zertifikate +You\ already\ added\ this\ certificate=Sie haben dieses Zertifikat bereits hinzugefügt Open\ folder=Ordner öffnen Export\ sort\ order=Sortierreihenfolge exportieren @@ -1279,6 +1299,7 @@ Please\ open\ %0\ manually.=Bitte öffnen Sie %0 manuell. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Der Link wurde in die Zwischenablage kopiert. Open\ %0\ file=%0 Datei öffnen +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Konsole konnte nicht automatisch erkannt werden. Bitte definieren Sie eine eigene Konsolenapplikation in den Einstellungen. Cannot\ delete\ file=Datei kann nicht gelöscht werden. File\ permission\ error=Fehler beim Dateizugriff. @@ -1666,6 +1687,8 @@ Issue\ report\ successful=Problemreport war erfolgreich Your\ issue\ was\ reported\ in\ your\ browser.=Ihr Problem wurde im Browser gemeldet. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Die Log und Ausnahmefehler-Informationen wurden in die Zwischenablage kopiert. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Bitte fügen Sie die Informationen (mit STRG+V) in die Problembeschreibung ein. +Last\ notification=Letzte Benachrichtigung +Check\ the\ event\ log\ to\ see\ all\ notifications=Überprüfe das Ereignisprotokoll, um alle Benachrichtigungen zu sehen Host=Host Port=Port @@ -1703,7 +1726,6 @@ Protected\ terms\ file=Geschützte Begriffsdatei Style\ file=Stildatei Open\ OpenOffice/LibreOffice\ connection=Öffne OpenOffice/LibreOffice Verbindung -You\ must\ enter\ at\ least\ one\ field\ name=Sie müssen mindestens einen Feldnamen angeben Non-ASCII\ encoded\ character\ found=Nicht ASCII-kodiertes Zeichen gefunden Non-UTF-8\ encoded\ field\ found=Nicht UTF-8 kodiertes Feld gefunden Toggle\ web\ search\ interface=Websuche ein-/ausschalten @@ -1841,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Entfernen aller Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Konnte Daten von '%0' nicht abrufen. Entry\ from\ %0\ could\ not\ be\ parsed.=Eintrag von %0 konnte nicht analysiert werden. Invalid\ identifier\:\ '%0'.=Ungültige Kennung\: "%0". -This\ paper\ has\ been\ withdrawn.=Dieses Paper wurde zurückgezogen. empty\ citation\ key=leerer Zitationsschlüssel Aux\ file=Aux-Datei Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Gruppe mit Einträgen die in einer vorhanden Tex-Datei zitiert worden sind @@ -2324,10 +2345,6 @@ Removes\ digits.=Ziffern entfernen. Presets=Voreinstellungen -Check\ Proxy\ Setting=Überprüfe Proxy-Einstellungen -Check\ connection=Verbindung prüfen -Connection\ failed\!=Verbindung fehlgeschlagen\! -Connection\ successful\!=Verbindung erfolgreich\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Gruppen aus Schlüsselwörtern im folgenden Feld generieren Generate\ groups\ for\ author\ last\ names=Gruppen für Nachnamen der Autoren generieren Regular\ expression=Regulärere Ausdruck @@ -2400,7 +2417,6 @@ Entry\ Type=Eintragstyp Entry\ types=Eintragstypen Field\ names=Feldnamen Others=Andere -Overwrite\ existing\ field\ values=Bestehende Feldwerte überschreiben Recommended=Empfohlen Authors\ and\ Title=Autoren und Titel @@ -2445,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Remote-Dienste Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Erlaube das Senden von PDF-Dateien und Rohzitaten an einen JabRef Online-Dienst (Grobid) um Metadaten zu ermitteln. Dies führt zu besseren Ergebnissen. +Fetcher\ cannot\ be\ tested\!=Der Fetcher kann nicht getestet werden\! +Fetcher\ unknown\!=Fetcher unbekannt\! Character\ by\ character=Zeichen für Zeichen Embedded=Eingebettet @@ -2468,26 +2486,32 @@ Valid\ from=Gültig seit Valid\ to=Gültig bis Signature\ algorithm=Signaturalgorithmus Version=Version -SSL\ Configuration=SSL-Konfiguration -SSL\ certificate\ file=SSL-Zertifikatsdatei -Duplicate\ Certificates=Doppelte Zertifikate -You\ already\ added\ this\ certificate=Sie haben dieses Zertifikat bereits hinzugefügt Error\ downloading=Fehler beim Herunterladen Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Fehler beim Schreiben von Metadaten. Details finden Sie im Fehlerprotokoll. Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metadaten können nicht geschrieben werden, Datei %1 nicht gefunden. Success\!\ Finished\ writing\ metadata.=Erfolgreich\! Das Schreiben der Metadaten ist abgeschlossen. -Overwrite\ Non\ empty\ fields=Overwrite Non empty fields -Set=Set -Append=Append -Clear\ field=Clear field -field\ value=field value -Edit\ field\ value\ of\ selected\ entries=Edit field value of selected entries -Rename=Rename -new\ field\ name=new field name -Copy\ value=Copy value -Move\ value=Move value -Swap\ values=Swap values -Copy\ or\ move\ the\ value\ of\ one\ field\ to\ another=Copy or move the value of one field to another +Custom\ API\ key=Eigener API-Schlüssel +Check\ %0\ API\ Key\ Setting=%0 API-Schlüsseleinstellungen überprüfen + +Edit\ content=Inhalt ändern +Copy\ or\ Move\ content=Inhalt kopieren oder verschieben +Overwrite\ field\ content=Inhalt des Feldes überschreiben +Set=Festlegen +Append=Anfügen +Clear\ field\ content=Feldinhalt löschen +Set\ or\ append\ content=Setze oder füge Inhalt an +Edit\ field\ content\ for\ selected\ entries=Feldinhalt der ausgewählten Einträge bearbeiten +Rename=Umbenennen +New\ field\ name=Neuer Feldname +Copy\ content=Inhalt kopieren +Move\ content=Inhalt verschieben +Swap\ content=Inhalte vertauschen +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Inhalt eines Feldes kopieren oder in ein anderes Feld verschieben +Automatic\ field\ editor=Automatischer Feldeditor + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Hinweis\: Wenn es den ursprünglichen Einträgen an Schlüsselwörtern fehlt, die sich für die neue Gruppenkonfiguration qualifizieren, wird die Bestätigung hier diese hinzufügen) +Assign=Zuweisen +Do\ not\ assign=Nicht zuweisen diff --git a/src/main/resources/l10n/JabRef_el.properties b/src/main/resources/l10n/JabRef_el.properties index 290a41f97b6..e3f04f56ffe 100644 --- a/src/main/resources/l10n/JabRef_el.properties +++ b/src/main/resources/l10n/JabRef_el.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 ταιριάζει την συνήθη έκφραση %1 %0\ matches\ the\ term\ %1=%0 ταιριάζει τον όρο %1 @@ -126,8 +127,6 @@ Cite\ command=Εντολή αναφοράς Clear=Καθαρισμός -Clear\ fields=Καθαρισμός πεδίων - Close\ dialog=Κλείσμο διαλόγου @@ -608,11 +607,18 @@ Remove\ selected\ entries\ from\ this\ group=Αφαίρεση επιλεγμέν Remove\ group=Αφαίρεση ομάδας -Remove\ group\ "%0"?=Αφαίρεση της ομάδας "%0"; +Remove\ group\ and\ subgroups=Αφαίρεση ομάδας και υπο-ομάδων + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Αφαίρεση της ομάδας "%0" και των υπο-ομάδων της; -Remove\ group\ and\ subgroups=Αφαίρεση ομάδας και υπο-ομάδων +Removed\ group\ "%0"\ and\ its\ subgroups.=Η ομάδα "%0" και οι υπο-ομάδες της έχουν αφαιρεθεί. + + Remove\ link=Αφαίρεση συνδέσμου @@ -621,8 +627,6 @@ Remove\ old\ entry=Αφαίρεση παλαιάς καταχώρησης Removed\ group\ "%0".=Η ομάδα "%0" έχει αφαιρεθεί. -Removed\ group\ "%0"\ and\ its\ subgroups.=Η ομάδα "%0" και οι υπο-ομάδες της έχουν αφαιρεθεί. - Removed\ string=Η συμβολοσειρά έχει αφαιρεθεί Renamed\ string=Η συμβολοσειρά έχει μετονομαστεί @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Επιλογή αρχείου από συμπι Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Επιλέξτε τους δενδροειδείς κόμβους για να προβάλλετε και να αποδεχτείτε ή να απορρίψετε αλλαγές -Set\ field=Ορισμός πεδίου -Set\ fields=Ορισμός πεδίων - Settings=Ρυθμίσεις Shortcut=Συντόμευση @@ -714,7 +715,6 @@ Size=Μέγεθος Skipped\ -\ No\ PDF\ linked=Παραβλέφθηκε - Δεν έχει συνδεθεί PDF Skipped\ -\ PDF\ does\ not\ exist=Παραβλέφθηκε - Δεν υπάρχει PDF -Skipped\ entry.=Η καταχώρηση παραβλέφθηκε. source\ edit=επεξεργασία πηγής @@ -829,10 +829,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru No\ full\ text\ document\ found=Δεν βρέθηκε αρχείο πλήρους κειμένου Download\ from\ URL=Κατέβασμα από σύνδεσμο URL Rename\ field=Μετονομασία πεδίου -Append\ field=Προσάρτηση πεδίου -Append\ to\ fields=Προσάρτηση στα πεδία -Rename\ field\ to=Μετονομασία πεδίου ως -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Μετακίνηση περιεχομένων ενός πεδίου σε ένα πεδίο με διαφορετικό όνομα Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Αδύνατη η χρήση της θύρας %0 για απομακρυσμένη λειτουργία, ίσως χρησιμοποιείται από μια άλλη εφαρμογή. Δοκιμάστε να ορίσετε άλλη θύρα. @@ -1046,6 +1042,7 @@ Proxy\ requires\ authentication=Ο διακομιστής μεσολάβησης Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Προσοχή\: Ο κωδικός πρόσβασης είναι αποθηκευμένος σε απλό κείμενο\! Clear\ connection\ settings=Εκκαθάριση ρυθμίσεων σύνδεσης + Open\ folder=Άνοιγμα φακέλου Export\ sort\ order=Εξαγωγή σειράς ταξινόμησης @@ -1448,7 +1445,6 @@ Protected\ terms\ file=Αρχείο προστατευμένων όρων Style\ file=Αρχείο στυλ Open\ OpenOffice/LibreOffice\ connection=Άνοιγμα σύνδεσης OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Πρέπει να καταχωρήσετε τουλάχιστον ένα όνομα πεδίου Non-ASCII\ encoded\ character\ found=Βρέθηκε χαρακτήρας χωρίς κωδικοποίηση ASCII Toggle\ web\ search\ interface=Εναλλαγή διεπαφής αναζήτησης στο διαδίκτυο @@ -1569,7 +1565,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Αφαιρεί Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Αδυναμία ανάκτησης δεδομένων καταχώρησης από '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Αδυναμία ανάλυσης καταχώρησης από %0. Invalid\ identifier\:\ '%0'.=Μη έγκυρο αναγνωριστικό\: '%0'. -This\ paper\ has\ been\ withdrawn.=Αυτή η εργασία έχει αποσυρθεί. Aux\ file=Αρχείο AUX Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Ομάδα που περιέχει καταχωρήσεις που αναφέρονται σε ένα συγκεκριμένο αρχείο TeX @@ -1702,6 +1697,8 @@ Default\ pattern=Προεπιλεγμένο μοτίβο + + diff --git a/src/main/resources/l10n/JabRef_en.properties b/src/main/resources/l10n/JabRef_en.properties index 5ae7dae2bab..ab98eefe526 100644 --- a/src/main/resources/l10n/JabRef_en.properties +++ b/src/main/resources/l10n/JabRef_en.properties @@ -833,7 +833,12 @@ Size=Size Skipped\ -\ No\ PDF\ linked=Skipped - No PDF linked Skipped\ -\ PDF\ does\ not\ exist=Skipped - PDF does not exist -Skipped\ entry.=Skipped entry. +JabRef\ skipped\ the\ entry.=JabRef skipped the entry. +Import\ error=Import error +Open\ library\ error=Open library error +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Please check your library file for wrong syntax. +SourceTab\ error=SourceTab error +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=User input via entry-editor in `{}bibtex source` tab led to failure. Sort\ subgroups=Sort subgroups @@ -2484,20 +2489,20 @@ Success\!\ Finished\ writing\ metadata.=Success! Finished writing metadata. Custom\ API\ key=Custom API key Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting -Edit\ field\ value=Edit field value -Two\ fields=Two fields -Overwrite\ Non\ empty\ fields=Overwrite Non empty fields +Edit\ content=Edit content +Copy\ or\ Move\ content=Copy or Move content +Overwrite\ field\ content=Overwrite field content Set=Set Append=Append -Clear\ field=Clear field -Field\ value=Field value -Edit\ field\ value\ of\ selected\ entries=Edit field value of selected entries +Clear\ field\ content=Clear field content +Set\ or\ append\ content=Set or append content +Edit\ field\ content\ for\ selected\ entries=Edit field content for selected entries Rename=Rename New\ field\ name=New field name -Copy\ value=Copy value -Move\ value=Move value -Swap\ values=Swap values -Copy\ or\ move\ the\ value\ of\ one\ field\ to\ another=Copy or move the value of one field to another +Copy\ content=Copy content +Move\ content=Move content +Swap\ content=Swap content +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copy or move the content of one field to another Automatic\ field\ editor=Automatic field editor Open\ Link=Open Link diff --git a/src/main/resources/l10n/JabRef_es.properties b/src/main/resources/l10n/JabRef_es.properties index c70107490a3..60e35d98ad0 100644 --- a/src/main/resources/l10n/JabRef_es.properties +++ b/src/main/resources/l10n/JabRef_es.properties @@ -1,7 +1,12 @@ +Could\ not\ delete\ empty\ entries.=No se pudieron eliminar entradas vacías. +Delete\ empty\ entries=Eliminar entradas vacías +Empty\ entries=Vaciar entradas +Keep\ empty\ entries=Mantener entradas vacías +Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=La biblioteca '%0' tiene entradas vacías. ¿Quieres eliminarlas? Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ restart.\ You\ may\ encounter\ errors\ if\ you\ continue\ with\ this\ session.=No es posible supervisar los cambios en los archivos. Cierre los archivos y procesos y reinicie. Puede que se produzcan errores si continúa con esta sesión. %0\ contains\ the\ regular\ expression\ %1=%0 contiene la expresión regular %1 @@ -15,6 +20,8 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entradas +Reveal\ in\ File\ Explorer=Mostrar en Explorador de Archivos + %0\ matches\ the\ regular\ expression\ %1=%0 coincidencias con la Expresión Regular %1 %0\ matches\ the\ term\ %1=%0 coincidencias con el término %1 @@ -48,6 +55,7 @@ The\ path\ need\ not\ be\ on\ the\ classpath\ of\ JabRef.=La ruta no debe estar Add\ a\ regular\ expression\ for\ the\ key\ pattern.=Añadir una expresión regular para el patrón clave. +Add\ entry\ manually=Añadir entrada manualmente Add\ selected\ entries\ to\ this\ group=Añadir entradas seleccionadas a este grupo @@ -144,8 +152,6 @@ Cite\ command=Comando Citar Clear=Limpiar -Clear\ fields=Limpiar campos - Open\ /\ close\ entry\ editor=Abrir o cerrar editor de entradas Close\ dialog=Cerrar diálogo @@ -167,6 +173,8 @@ Copy=Copiar Copy\ title=Copiar título Copy\ \\cite{citation\ key}=Copiar \\cite{citation key} +Copy\ citation\ (html)=Copiar cita (html) +Copy\ citation\ (text)=Copiar cita (texto) Copy\ citation\ key=Copiar clave de cita Copy\ citation\ key\ and\ link=Copiar clave de cita y enlace Copy\ citation\ key\ and\ title=Copiar clave y título de cita @@ -215,13 +223,17 @@ cut\ entries=Cortar entradas cut\ entry\ %0=corte de entrada %0 +DOI\ not\ found=DOI no encontrado Library\ encoding=Codificación de la biblioteca Library\ properties=Propiedades de la biblioteca +%0\ -\ Library\ properties=%0 - Propiedades de la biblioteca Default=Por defecto +Character\ encoding\ UTF-8\ is\ not\ supported.=La codificación de caracteres UTF-8 no restá soportada. +UTF-8\ could\ not\ be\ used\ to\ encode\ the\ following\ characters\:\ %0=UTF-8 no se pudo utilizar para codificar los siguientes caracteres\: %0 The\ chosen\ encoding\ '%0'\ could\ not\ encode\ the\ following\ characters\:=La codificación de caracteres '%' no puede codificar los siguientes caracteres\: Downloading=Descargando @@ -270,6 +282,7 @@ Downloaded\ website\ as\ an\ HTML\ file.=Se descargó el sitio web como un archi duplicate\ removal=eliminación de duplicados +Duplicate\ fields=Campos duplicados Duplicate\ string\ name=Nombre de cadena duplicado @@ -331,6 +344,7 @@ External\ file\ links=Enlaces a archivos externos External\ programs=Programas externos +Failed\ to\ import\ by\ ID=Error al importar por ID Field=Campo @@ -355,6 +369,7 @@ Filter=Filtro Filter\ groups=Filtros +Finished\ writing\ metadata\ for\ %0\ file\ (%1\ skipped,\ %2\ errors).=Escritura de metadatos para el archivo %0 finalizada (%1 omitidos, %2 errores). First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=En primer lugar, seleccione las entradas para las que desea generar claves @@ -371,6 +386,7 @@ Formatter\ name=Nombre del formateador found\ in\ AUX\ file=encontrado en archivo AUX +Fulltext\ search=Búsqueda de texto completo Fulltext\ for=Texto completo de @@ -449,6 +465,9 @@ Include\ subgroups\:\ When\ selected,\ view\ entries\ contained\ in\ this\ group Independent\ group\:\ When\ selected,\ view\ only\ this\ group's\ entries=Grupo independiente\: ver sólo las entradas de este grupo cuando esté seleccionado. I\ Agree=Acepto +Indexing\ pdf\ files=Indexando archivos pdf +Indexing\ for\ %0=Indexando para %0 +%0\ of\ %1\ linked\ files\ added\ to\ the\ index=%0 de %1 archivos enlazados añadidos al índice Invalid\ citation\ key=La clave de cita no es válida @@ -462,6 +481,8 @@ JabRef\ requests\ recommendations\ from\ Mr.\ DLib,\ which\ is\ an\ external\ se JabRef\ Version\ (Required\ to\ ensure\ backwards\ compatibility\ with\ Mr.\ DLib's\ Web\ Service)=Versión JabRef (necesaria para asegurar la compatibilidad con el servicio web de Mr. DLib) Journal\ abbreviations=Abreviaturas de publicaciones +Journal\ lists\:=Listados de revistas\: +Remove\ journal\ '%0'=Eliminar revista '%0' Keep\ both=Mantener ambos @@ -497,6 +518,7 @@ Main\ file\ directory=Carpeta del archivo principal Manage\ custom\ exports=Administrar exportaciones personalizadas Manage\ custom\ imports=Administrar importaciones personalizadas +External\ file\ types=Tipos de archivos externos Mark\ new\ entries\ with\ owner\ name=Marcar nuevas entradas con nombre de propietario @@ -519,7 +541,9 @@ Moved\ group\ "%0".=Se ha movido el grupo "%0". Mr.\ DLib\ Privacy\ settings=Mr. DLib Configuración de la privacidad +No\ database\ is\ open=No hay ninguna base de datos abierta +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=Necesitamos una base de datos desde la que exportar. Abrir una. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=No se han recibido recomendaciones del Mr. DLib para esta entrada. @@ -553,6 +577,7 @@ No\ journal\ names\ could\ be\ abbreviated.=No se pudieron abreviar nombres de r No\ journal\ names\ could\ be\ unabbreviated.=No se pudieron expandir nombres de revistas. +No\ DOI\ data\ exists=No existen datos DOI not=no @@ -677,11 +702,25 @@ Remove\ selected\ entries\ from\ this\ group=Eliminar las entradas seleccionadas Remove\ group=Eliminar grupo -Remove\ group\ "%0"?=¿Eliminar el grupo "%0%? +Remove\ group\ and\ subgroups=Eliminar grupo y subgrupos + +Remove\ groups\ and\ subgroups=Eliminar grupos y subgrupos + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=¿Eliminar todos los grupos seleccionados y mantener sus subgrupos? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=¿Eliminar grupo "%0" y mantener sus subgrupos? + +Remove\ groups=Eliminar grupos + +Removed\ all\ selected\ groups.=Eliminados todos los grupos seleccionados. Remove\ group\ "%0"\ and\ its\ subgroups?=¿Eliminar el grupo "%0" y sus subgrupos? -Remove\ group\ and\ subgroups=Eliminar grupo y subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Se ha eliminado el grupo "%0" y sus subgrupos. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=¿Eliminar todos los grupos seleccionados y sus subgrupos? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Eliminados todos los grupos seleccionados y sus subgrupos. Remove\ link=Eliminar enlace @@ -691,8 +730,6 @@ Remove\ string\ %0=Eliminar cadena %0 Removed\ group\ "%0".=el grupo "%0" se ha eliminado. -Removed\ group\ "%0"\ and\ its\ subgroups.=Se ha eliminado el grupo "%0" y sus subgrupos. - Removed\ string=Cadena eliminada Renamed\ string=Cadena renombrada @@ -715,6 +752,8 @@ Replaces\ Unicode\ ligatures\ with\ their\ expanded\ form=Reemplaza las ligadura Required\ fields=Campos requeridos +Do\ not\ resolve\ BibTeX\ strings=No resolver cadenas BibTeX +Resolve\ BibTeX\ strings\ for\ the\ following\ fields=Resolver las cadenas BibTeX para los siguientes campos resolved=resuelto @@ -722,11 +761,13 @@ Restart=Reiniciar Restart\ required=Reinicio requerido +Return\ to\ dialog=Volver al diálogo Review=Revisar Review\ changes=Revisar cambios Review\ Field\ Migration=Revisar campo de migración +Loading=Cargando Save=Guardar Save\ all\ finished.=Guardar todos los finalizados @@ -740,6 +781,7 @@ Save\ library\ as...=Guardar biblioteca como... Saving=Guardando Saving\ all\ libraries...=Guardando todas las bibliotecas... Saving\ library=Guardando biblioteca +Library\ saved=Biblioteca guardada Saved\ selected\ to\ '%0'.=Selección guardada en '%0'. Search=Buscar @@ -759,9 +801,6 @@ Select\ file\ from\ ZIP-archive=Seleccionar archivo desde archivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Seleccionar nodos de árbol para ver y aceptar o rechazar los cambios. -Set\ field=Establecer campo -Set\ fields=Establecer campos - Settings=Ajustes Shortcut=Atajo @@ -794,7 +833,12 @@ Size=Tamaño Skipped\ -\ No\ PDF\ linked=Omitido - No se enlazó PDF Skipped\ -\ PDF\ does\ not\ exist=Omitido - No existe el PDF -Skipped\ entry.=Entrada omitida. +JabRef\ skipped\ the\ entry.=JabRef omitió la entrada. +Import\ error=Error al importar +Open\ library\ error=Error al abrir la biblioteca +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Por favor, comprueba errores de sintaxis en tu archivo de biblioteca. +SourceTab\ error=Error de SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Entrada de usuario a través del editor de entrada en la pestaña `{}bibtex source` condujo al error. Sort\ subgroups=Ordenar subgrupos @@ -883,13 +927,18 @@ Warning=Advertencia Warnings=Advertencias +Warning\:\ You\ added\ field\ "%0"\ twice.\ Only\ one\ will\ be\ kept.=Advertencia\: Añadiste el campo "%0" dos veces. Solo uno será utilizado. web\ link=enlace a web What\ do\ you\ want\ to\ do?=¿Qué desea hacer? Whatever\ option\ you\ choose,\ Mr.\ DLib\ may\ share\ its\ data\ with\ research\ partners\ to\ further\ improve\ recommendation\ quality\ as\ part\ of\ a\ 'living\ lab'.\ Mr.\ DLib\ may\ also\ release\ public\ datasets\ that\ may\ contain\ anonymized\ information\ about\ you\ and\ the\ recommendations\ (sensitive\ information\ such\ as\ metadata\ of\ your\ articles\ will\ be\ anonymised\ through\ e.g.\ hashing).\ Research\ partners\ are\ obliged\ to\ adhere\ to\ the\ same\ strict\ data\ protection\ policy\ as\ Mr.\ DLib.=Sea cual sea la opción que elija, Mr. DLib puede compartir sus datos con socios de investigación para mejorar aún más la calidad de las recomendaciones como parte de un "laboratorio vivo". Mr. DLib también puede publicar conjuntos de datos públicos que pueden contener información anónima sobre usted y las recomendaciones (la información confidencial como los metadatos de sus artículos será anonimizada a través de, por ejemplo, hashing). Los socios de investigación están obligados a adherirse a la misma estricta política de protección de datos que Mr. DLib. +Will\ write\ metadata\ to\ the\ PDFs\ linked\ from\ selected\ entries.=Escribirá metadatos en los PDFs enlazados desde entradas seleccionadas. +Write\ BibTeXEntry\ as\ metadata\ to\ PDF.=Escribir entrada BibTeX como metadatos XMP en el PDF. +Write\ metadata\ for\ all\ PDFs\ in\ current\ library?=¿Escribir metadatos para todos los PDFs de la biblioteca actual? +Writing\ metadata\ for\ selected\ entries...=Escribiendo metadatos para las entradas seleccionadas... Write\ BibTeXEntry\ as\ XMP\ metadata\ to\ PDF.=Escribe BibTeXEntry como metadatos XMP en los PDF. @@ -929,10 +978,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Renglón %0\: No\ full\ text\ document\ found=No se encontró un documento de texto completo Download\ from\ URL=Descargar desde URL Rename\ field=Renombrar campo -Append\ field=Añadir campo -Append\ to\ fields=Añadir a campos -Rename\ field\ to=Renombrar campo a -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover contenidos de un campo en un campo con nombre diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=No se puede usar el puerto %0 para operación remota\: Puede estar en uso por otra aplicación. Pruebe a especificar otro puerto. @@ -1182,6 +1227,11 @@ Use\ custom\ proxy\ configuration=Usar configuración de proxy personalizada Proxy\ requires\ authentication=El proxi requiere autenticación Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atención\: ¡La contraseña está almacenada como texto plano\! Clear\ connection\ settings=Limpiar ajustes de conexión +Check\ Proxy\ Setting=Comprobar configuración de «proxy» +Check\ connection=Comprobar conexión +Connection\ failed\!=Falló la conexión. +Connection\ successful\!=Conexión correcta. + Open\ folder=Abrir pasta Export\ sort\ order=Criterio de ordenación para exportación @@ -1586,6 +1636,8 @@ Issue\ report\ successful=Comunicación de problema satisfactoria Your\ issue\ was\ reported\ in\ your\ browser.=Su problema fue comunicado a través del navegador The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=La información de excepción y registro fue copiada a su portapapeles Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Por favor, pegue esta información (con Ctrl+V) en la descripción del problema. +Last\ notification=Última notificación +Check\ the\ event\ log\ to\ see\ all\ notifications=Revisa el registro de eventos para ver todas las notificaciones Host=Host/Servidor Port=Puerto @@ -1623,7 +1675,6 @@ Protected\ terms\ file=Archivo de términos protegidos Style\ file=Archivo de estilo Open\ OpenOffice/LibreOffice\ connection=Conexión OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Debe introducir, al menos, un nombre de campo Non-ASCII\ encoded\ character\ found=Se han encontrado caracteres con codificación no-ASCII Non-UTF-8\ encoded\ field\ found=Se encontró un campo no codificado en UTF-8 Toggle\ web\ search\ interface=Cambiar interfaz de búsqueda web @@ -1732,6 +1783,7 @@ Delete\ '%0'=Eliminar '%0' Delete\ from\ disk=Eliminar del disco duro Remove\ from\ entry=Eliminar de la entrada There\ exists\ already\ a\ group\ with\ the\ same\ name.=Ya existe un grupo con el mismo nombre. +If\ you\ use\ it,\ it\ will\ inherit\ all\ entries\ from\ this\ other\ group.=Si lo utiliza, heredará todas las entradas de este otro grupo. Copy\ linked\ file=Copiar archivo enlazado Copy\ linked\ file\ to\ folder...=Copiar archivo enlazado a la carpeta... @@ -1760,7 +1812,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Elimina todos l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=No se pueden recuperar los datos de la entrada desde '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=No se pudo analizar la entrada de %0. Invalid\ identifier\:\ '%0'.=Identificador no válido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Este artículo ha sido retirado. empty\ citation\ key=clave de cita vacía Aux\ file=Archivo Aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo que contiene las entradas citadas en un archivo TeX determinado @@ -1788,6 +1839,7 @@ Could\ not\ connect\ to\ Vim\ server.\ Make\ sure\ that\ Vim\ is\ running\ with\ Could\ not\ connect\ to\ a\ running\ gnuserv\ process.\ Make\ sure\ that\ Emacs\ or\ XEmacs\ is\ running,\ and\ that\ the\ server\ has\ been\ started\ (by\ running\ the\ command\ 'server-start'/'gnuserv-start').=No se puede conectar con un proceso gnuserv en ejecución. Asegúrese de que Emacs o XEmacs se está ejecutando y de que el servidor ha sido iniciado (ejecutando el comando 'server-start'/'gnuserv-start'). Error\ pushing\ entries=Error al enviar entradas +Preamble=Preámbulo Markings=Marcados Use\ selected\ instance=Usar la instancia seleccionada @@ -1806,6 +1858,7 @@ Blog=Blog Check\ integrity=Verificar integridad Cleanup\ URL\ link=Limpiar un enlace URL Cleanup\ URL\ link\ by\ removing\ special\ symbols\ and\ extracting\ simple\ link=Limpiar un enlace URL eliminando los símbolos especiales y extrayendo un enlace simple +Copy\ DOI=Copiar DOI Copy\ DOI\ url=Copiar la url del DOI Development\ version=Versión de desarrollo Export\ selected\ entries=Exportar registros seleccionados @@ -1815,6 +1868,8 @@ JabRef\ resources=Recursos sobre JabRef Manage\ journal\ abbreviations=Administrar abreviaturas de publicaciones Manage\ protected\ terms=gestionar términos protegidos New\ entry\ from\ plain\ text=Nueva entrada desde texto sin formato +Import\ by\ ID=Importar por ID +Enter\ a\ valid\ ID=Introduzca un ID válido New\ sublibrary\ based\ on\ AUX\ file=Nueva subbiblioteca a partir de un archivo AUX Push\ entries\ to\ external\ application\ (%0)=Agregar registros a aplicación externa (%0) Quit=Salir @@ -1874,6 +1929,7 @@ Keyword\ separator=Separador de palabras clave Remove\ keyword=Eliminar palabra clave Are\ you\ sure\ you\ want\ to\ remove\ keyword\:\ "%0"?=¿Está seguro de que desea eliminar la palabra clave\: "%0"? Reset\ to\ default=Restablecer los valores por defecto +String\ constants=Constantes de cadena Export\ all\ entries=Exportar todas las entradas Generate\ citation\ keys=Generar claves de cita Manage\ field\ names\ &\ content=Gestionar nombres y contenido de los campos @@ -2003,6 +2059,7 @@ Please\ provide\ a\ valid\ aux\ file.=Por favor, proporcione un archivo AUX vál Keyword\ delimiter=Separador de palabras clave Hierarchical\ keyword\ delimiter=Separador de palabras clave jerárquicas Escape\ ampersands=Escape de ampersands +Escape\ dollar\ sign=Símbolo de escape dólar Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Consejo\:\n\nPara buscar Pedro en todos los campos, escriba\:\npedro\n\nPara buscar Pedro en el campo author y eléctrico en el campo title, escriba\:\nauthor\=Pedro and title\=eléctrico @@ -2212,7 +2269,11 @@ Reveal\ in\ file\ explorer=Revelar en el explorador de archivos Autolink\ files=Enlazar archivos automáticamente +Custom\ editor\ tabs=Pestañas de editor personalizadas +Custom\ export\ formats=Formatos de exportación personalizados +Custom\ import\ formats=Formatos de importación personalizados +No\ list\ enabled=No hay lista habilitada Protect\ selection=Proteger selección Customized\ preview\ style=Estilo de previsualización personalizado @@ -2233,17 +2294,23 @@ Removes\ digits.=Elimina dígitos. Presets=Preconfiguraciones -Check\ Proxy\ Setting=Comprobar configuración de «proxy» -Check\ connection=Comprobar conexión -Connection\ failed\!=Falló la conexión. -Connection\ successful\!=Conexión correcta. Generate\ groups\ from\ keywords\ in\ the\ following\ field=Generar grupos a partir de palabras clave en el campo siguiente Generate\ groups\ for\ author\ last\ names=Generar grupos para apellidos de autores Regular\ expression=Expresión regular Error\ importing.\ See\ the\ error\ log\ for\ details.=Error al importar. Consulte el registro de errores para obtener detalles. +Error\ from\ import\:\ %0=Error de importación\: %0 +Error\ reading\ PDF\ content\:\ %0=Error al leer el contenido PDF\: %0 +Importing\ bib\ entry=Importando entrada bibliográfica +Importing\ using\ extracted\ PDF\ data=Importación usando datos extraídos del PDF +No\ BibTeX\ data\ found.\ Creating\ empty\ entry\ with\ file\ link=No se encontraron datos BibTeX. Creando entrada vacía con enlace de archivo +No\ metadata\ found.\ Creating\ empty\ entry\ with\ file\ link=No se encontraron metadatos. Creando entrada vacía con enlace de archivo +Processing\ file\ %0=Procesando archivo %0 +Export\ selected=Exportar seleccionados +Separate\ merged\ citations=Citas fusionadas separadas +Separate\ citations=Citas separadas Custom\ DOI\ URI=URI personalizado de DOI @@ -2272,7 +2339,6 @@ Entry\ Type=Tipo de entrada Entry\ types=Tipos de entrada Field\ names=Nombre del campo Others=Otros -Overwrite\ existing\ field\ values=Sobreescribir valores de campo existentes Recommended=Recomendado Authors\ and\ Title=Autoría y título @@ -2303,3 +2369,5 @@ Question=Pregunta + + diff --git a/src/main/resources/l10n/JabRef_fa.properties b/src/main/resources/l10n/JabRef_fa.properties index 9fabe5b9655..fb60c5076e1 100644 --- a/src/main/resources/l10n/JabRef_fa.properties +++ b/src/main/resources/l10n/JabRef_fa.properties @@ -14,6 +14,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ + Abbreviate\ names=نام های مخفف Abbreviated\ %0\ journal\ names.=نام های مخفف شده %0 ژورنال. @@ -98,8 +99,6 @@ Case\ sensitive=حساس به کوچکی و بزرگی حروف Clear=پاکسازی -Clear\ fields=پاک کردن فیلدها - @@ -375,13 +374,18 @@ Open\ terminal\ here=در اینجا پایانه را باز کن -Library\ saved=کتابخانه ذخیره شد +Library\ saved=کتابخانه ذخیره شد + + + + + @@ -496,6 +500,7 @@ Merge\ entries=ترکیب کردن ورودیها + Open\ folder=بازکردن پوشه @@ -665,3 +670,5 @@ Search\ results\ from\ open\ libraries=جستجوی نتایج از کتابخا + + diff --git a/src/main/resources/l10n/JabRef_fr.properties b/src/main/resources/l10n/JabRef_fr.properties index 425843db382..46481f71dce 100644 --- a/src/main/resources/l10n/JabRef_fr.properties +++ b/src/main/resources/l10n/JabRef_fr.properties @@ -19,7 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entrées -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=Exportation de %0 réussie. Voulez-vous ouvrir le dossier contenant le fichier sauvegardé ? +Export\ operation\ finished\ successfully.=L'opération d'export s'est terminée avec succès. + +Reveal\ in\ File\ Explorer=Montrer dans l'explorateur de fichiers %0\ matches\ the\ regular\ expression\ %1=%0 correspond à l'expression régulière %1 @@ -151,8 +153,6 @@ Cite\ command=Commande Cite Clear=Vider -Clear\ fields=Vider les champs - Open\ /\ close\ entry\ editor=Ouvrir/fermer l'éditeur d'entrées Close\ dialog=Fermer la fenêtre @@ -295,7 +295,7 @@ Dynamically\ group\ entries\ by\ searching\ a\ field\ for\ a\ keyword=Grouper dy Each\ line\ must\ be\ of\ the\ following\ form\:\ 'tab\:field1;field2;...;fieldN'.=Chaque ligne doit être au format suivant \: 'tab\:field1;field2;...;fieldN'. -Edit=Éditer +Edit=Édition Edit\ file\ type=Modifier le type de fichier @@ -703,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Supprimer de ce groupe les entrées Remove\ group=Supprimer le groupe -Remove\ group\ "%0"?=Supprimer le groupe « %0 » ? +Remove\ group\ and\ subgroups=Supprimer le groupe et les sous-groupes + +Remove\ groups\ and\ subgroups=Supprimer les groupes et les sous-groupes + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Supprimer tous les groupes sélectionnés et conserver leurs sous-groupes ? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Supprimer le groupe « %0 » et conserver ses sous-groupes ? + +Remove\ groups=Supprimer les groupes + +Removed\ all\ selected\ groups.=Tous les groupes sélectionnés ont été supprimés. Remove\ group\ "%0"\ and\ its\ subgroups?=Supprimer le groupe « %0 » et ses sous-groupes ? -Remove\ group\ and\ subgroups=Supprimer le groupe et les sous-groupes +Removed\ group\ "%0"\ and\ its\ subgroups.=Groupe « %0 » et ses sous-groupes supprimés. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Supprimer tous les groupes sélectionnés et leurs sous-groupes ? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Tous les groupes sélectionnés et leurs sous-groupes ont été supprimés. Remove\ link=Supprimer le lien @@ -717,8 +731,6 @@ Remove\ string\ %0=Supprimer la chaîne %0 Removed\ group\ "%0".=Groupe « %0 » supprimé. -Removed\ group\ "%0"\ and\ its\ subgroups.=Groupe « %0 » et ses sous-groupes supprimés. - Removed\ string=Chaîne supprimée Renamed\ string=Chaîne renommée @@ -770,7 +782,7 @@ Save\ library\ as...=Enregistrer le fichier sous... Saving=Enregistrement en cours Saving\ all\ libraries...=Enregistrement de tous les fichiers... Saving\ library=Enregistrement du fichier en cours -Library\ saved=Fichier sauvé +Library\ saved=Fichier enregistré Saved\ selected\ to\ '%0'.=Sélection enregistrée dans « %0 ». Search=Recherche @@ -790,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Sélectionner un fichier depuis une archive ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Sélectionner les nœuds de l'arborescence pour voir, et accepter ou rejeter, les modifications -Set\ field=Configurer le champ -Set\ fields=Configurer les champs - Settings=Paramètres Shortcut=Raccourci @@ -825,7 +834,12 @@ Size=Taille Skipped\ -\ No\ PDF\ linked=Sauté - Pas de PDF lié Skipped\ -\ PDF\ does\ not\ exist=Omis - Le PDF n'existe pas -Skipped\ entry.=Entrée omise. +JabRef\ skipped\ the\ entry.=JabRef a ignoré l'entrée. +Import\ error=Erreur d'importation +Open\ library\ error=Erreur d'ouverture du fichier +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Veuillez vérifier la syntaxe de votre fichier bibliographique. +SourceTab\ error=Erreur de SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=La saisie de l'utilisateur via l'éditeur d'entrée dans l'onglet `{}bibtex source` a conduit à un échec. Sort\ subgroups=Trier les sous-groupes @@ -971,18 +985,14 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Ligne Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Ligne %0 \: clef de citation corrompue %1 (virgule manquante). No\ full\ text\ document\ found=Aucun texte intégral trouvé Download\ from\ URL=Télécharger depuis l'URL -Rename\ field=Renommer le champ -Append\ field=Ajouter au champ -Append\ to\ fields=Ajouter aux champs -Rename\ field\ to=Renommer le champ en -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Déplacer le contenu d'un champ vers un champ d'un nom différent +Rename\ field=Renommer Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Le port %0 ne peut pas être utilisé pour une opération à distance ; un autre logiciel pourrait être en train de l'utiliser. Essayer de spécifier un autre port. Looking\ for\ full\ text\ document...=Recherche du texte intégral... A\ local\ copy\ will\ be\ opened.=Une copie locale sera ouverte. -Autosave\ local\ libraries=Sauvegarder automatiquement les fichiers locaux -Automatically\ save\ the\ library\ to=Sauvegarder automatiquement les fichiers dans \: +Autosave\ local\ libraries=Enregistrer automatiquement les fichiers locaux +Automatically\ save\ the\ library\ to=Enregistrer automatiquement les fichiers dans \: Please\ enter\ a\ valid\ file\ path.=Veuillez saisir un chemin de fichier valide. Error\ opening\ file\ '%0'.=Erreur lors de l'ouverture du fichier « %0 ». @@ -1003,7 +1013,7 @@ contains=contient search\ expression=expression de recherche Optional\ fields\ 2=Champs optionnels 2 -Waiting\ for\ save\ operation\ to\ finish=Attente de la fin de l'opération de sauvegarde +Waiting\ for\ save\ operation\ to\ finish=Attente de la fin de l'opération d'enregistrement Waiting\ for\ background\ tasks\ to\ finish.\ Quit\ anyway?=En attente de la fin des tâches en arrière-plan. Quitter quand même ? Find\ and\ remove\ duplicate\ citation\ keys=Rechercher et supprimer les clefs de citation dupliquées @@ -1037,7 +1047,7 @@ Output\ file\ missing=Fichier de sortie manquant No\ search\ matches.=Recherche sans correspondance. The\ output\ option\ depends\ on\ a\ valid\ input\ option.=L'option de sortie dépend d'une option d'entrée valide. Linked\ file\ name\ conventions=Conventions pour les noms de fichiers liés -Filename\ format\ pattern=Modèle de format de nom de fichier +Filename\ format\ pattern=Modèle de format de nom de fichier Additional\ parameters=Paramètres additionnels Cite\ selected\ entries\ between\ parenthesis=Citer les entrées sélectionnées entre parenthèses Cite\ selected\ entries\ with\ in-text\ citation=Citer les entrées sélectionnées comme incluse dans le texte @@ -1243,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Utiliser une configuration de proxy personnali Proxy\ requires\ authentication=Le proxy nécessite une authentification Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Attention \: Le mot de passe est stocké en clair dans du texte brut \! Clear\ connection\ settings=Réinitialiser les paramètres de connexion +Check\ Proxy\ Setting=Vérifier la configuration du proxy +Check\ connection=Vérifier la connexion +Connection\ failed\!=Échec de la connexion \! +Connection\ successful\!=Connexion réussie \! + +SSL\ Configuration=Configuration SSL +SSL\ configuration\ changed=Configuration SSL modifiée +SSL\ certificate\ file=Fichier de certificat SSL +Duplicate\ Certificates=Dupliquer les certificats +You\ already\ added\ this\ certificate=Vous avez déjà ajouté ce certificat Open\ folder=Ouvrir le répertoire Export\ sort\ order=Exporter l'ordre de tri @@ -1279,6 +1299,7 @@ Please\ open\ %0\ manually.=Veuillez ouvrir manuellement %0 . The\ link\ has\ been\ copied\ to\ the\ clipboard.=Le lien a été copié dans le presse-papiers. Open\ %0\ file=Ouvrir le fichier %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Impossible de détecter le terminal automatiquement. Veuillez définir un terminal personnalisé dans les préférences. Cannot\ delete\ file=Le fichier ne peut pas être supprimé File\ permission\ error=Erreur due aux permissions du fichier @@ -1666,6 +1687,8 @@ Issue\ report\ successful=Signalement de l'anomalie réussi Your\ issue\ was\ reported\ in\ your\ browser.=Votre anomalie a été affichée dans votre navigateur. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Le journal et les informations d'anomalie ont été copiées dans votre presse-papier. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Veuillez coller ces informations (avec Ctrl+V) dans la description de l'anomalie. +Last\ notification=Dernière notification +Check\ the\ event\ log\ to\ see\ all\ notifications=Consultez le journal des événements pour voir toutes les notifications Host=Hôte Port=Port @@ -1703,7 +1726,6 @@ Protected\ terms\ file=Fichier des termes protégés Style\ file=Fichier de style Open\ OpenOffice/LibreOffice\ connection=Ouvrir une connexion OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Vous devez saisir au moins un nom de champ Non-ASCII\ encoded\ character\ found=Caractère ayant un encodage non-ASCII trouvé Non-UTF-8\ encoded\ field\ found=Un champ non encodé en UTF-8 a été trouvé Toggle\ web\ search\ interface=Afficher/Masquer l'interface de recherche internet @@ -1841,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Supprime du con Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Impossible de récupérer les données sur l'entrée depuis « %0 ». Entry\ from\ %0\ could\ not\ be\ parsed.=Entrée de %0 n’a pas pu être analysée. Invalid\ identifier\:\ '%0'.=Identifiant non valide \: « %0 ». -This\ paper\ has\ been\ withdrawn.=Cet article a été retiré. empty\ citation\ key=clef de citation vide Aux\ file=Fichier AUX Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Groupe contenant les entrées citées dans un fichier TeX spécifique @@ -2144,7 +2165,7 @@ Search\ ShortScience=Recherche ShortScience Unable\ to\ open\ ShortScience.=Impossible de se connecter à ShortScience. Shared\ database=Base de données partagée -Lookup=Rechercher +Lookup=Recherche Please\ enter\ a\ field\ name\ to\ search\ for\ a\ keyword.=Veuillez entrer le nom du champ dans lequel chercher un mot-clef. Access\ date\ of\ the\ address\ specified\ in\ the\ url\ field.=Date d'accès de l'adresse spécifiée dans le champ "url". @@ -2324,10 +2345,6 @@ Removes\ digits.=Supprime les chiffres. Presets=Préréglages -Check\ Proxy\ Setting=Vérifier la configuration du proxy -Check\ connection=Vérifier la connexion -Connection\ failed\!=Échec de la connexion \! -Connection\ successful\!=Connexion réussie \! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Créer des groupes à partir des mot-clefs du champ suivant Generate\ groups\ for\ author\ last\ names=Créer des groupes à partir des noms propres des auteurs Regular\ expression=Expression régulière @@ -2400,7 +2417,6 @@ Entry\ Type=Type d’entrée Entry\ types=Types d'entrée Field\ names=Noms de champs Others=Autres -Overwrite\ existing\ field\ values=Écraser les valeurs des champs existants Recommended=Usuels Authors\ and\ Title=Auteurs et titre @@ -2470,10 +2486,6 @@ Valid\ from=Valable à partir de Valid\ to=Valide jusqu'au Signature\ algorithm=Algorithme de signature Version=Version -SSL\ Configuration=Configuration SSL -SSL\ certificate\ file=Fichier de certificat SSL -Duplicate\ Certificates=Dupliquer les certificats -You\ already\ added\ this\ certificate=Vous avez déjà ajouté ce certificat Error\ downloading=Erreur de téléchargement @@ -2483,3 +2495,23 @@ Success\!\ Finished\ writing\ metadata.=Succès \! Écriture des métadonnées t Custom\ API\ key=Clef d'API personnalisée Check\ %0\ API\ Key\ Setting=Vérifier les paramètres de la clef d'API %0 + +Edit\ content=Modifier le contenu +Copy\ or\ Move\ content=Copier ou déplacer le contenu +Overwrite\ field\ content=Écraser le contenu du champ +Set=Définir +Append=Ajouter +Clear\ field\ content=Effacer le contenu du champ +Set\ or\ append\ content=Définir ou ajouter du contenu +Edit\ field\ content\ for\ selected\ entries=Modifier le contenu d'un champ +Rename=Renommer +New\ field\ name=Nouveau nom de champ +Copy\ content=Copier le contenu +Move\ content=Déplacer le contenu +Swap\ content=Permuter le contenu +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copier ou déplacer le contenu d'un champ vers un autre +Automatic\ field\ editor=Éditeur automatique de champs + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note \: si les entrées originales n'ont pas de mots-clefs correspondant à la nouvelle configuration du groupe, confirmer ici les ajoutera) +Assign=Assigner +Do\ not\ assign=Ne pas assigner diff --git a/src/main/resources/l10n/JabRef_id.properties b/src/main/resources/l10n/JabRef_id.properties index 0e582b7cb21..6458e06d8e0 100644 --- a/src/main/resources/l10n/JabRef_id.properties +++ b/src/main/resources/l10n/JabRef_id.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 sesuai dengan Ekspresi Reguler %1 %0\ matches\ the\ term\ %1=%0 sesuai dengan istilah %1 @@ -125,8 +126,6 @@ Cite\ command=Perintah acuan Clear=Bersihkan -Clear\ fields=Bersihkan beberapa bidang - Close\ dialog=Tutup dialog @@ -604,11 +603,18 @@ Remove\ selected\ entries\ from\ this\ group=Menghapus entri yang dipilih dari g Remove\ group=Hapus grup -Remove\ group\ "%0"?=Hapus grup "%0"? +Remove\ group\ and\ subgroups=Hapus grup dan sub-grup + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Hapus grup "%0" dan sub-grup nya? -Remove\ group\ and\ subgroups=Hapus grup dan sub-grup +Removed\ group\ "%0"\ and\ its\ subgroups.=Hapus grup "%0" dan sub-grup nya. + + Remove\ link=Hapus tautan @@ -617,8 +623,6 @@ Remove\ old\ entry=Hapus entri lama Removed\ group\ "%0".=Hapus grup "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Hapus grup "%0" dan sub-grup nya. - Removed\ string=Hapus string Renamed\ string=Ganti nama string @@ -672,9 +676,6 @@ Select\ file\ from\ ZIP-archive=Pilih berkas dari arsip ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Pilih tiga nodal untuk melihat, menerima atau menolak perubahan -Set\ field=Pilih bidang -Set\ fields=Pilih beberapa bidang - Settings=Pengaturan Shortcut=Pintasan @@ -705,7 +706,6 @@ Size=Ukuran Skipped\ -\ No\ PDF\ linked=Dilompati - Tanpa tautan PDF Skipped\ -\ PDF\ does\ not\ exist=Dilompati - PDF tidak ada -Skipped\ entry.=Entri dilompati. source\ edit=sunting sumber @@ -819,10 +819,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Unduh dari URL Rename\ field=Ganti nama bidang -Append\ field=Tambahkan bidang -Append\ to\ fields=Tambahkan ke bidang -Rename\ field\ to=Ganti nama bidang menjadi -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Pindah isi dari bidang ke bidang lain dengan nama lain Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Tidak bisa memakai port %0 untuk operasi jauh; aplikasi lain mungkin sedang menggunakan. Coba port lain. @@ -1036,6 +1032,7 @@ Proxy\ requires\ authentication=Proxy memerlukan otentikasi Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Perhatian\: Kata sandi disimpan sebagai teks biasa\! Clear\ connection\ settings=Bersihkan pengaturan koneksi + Open\ folder=Buka direktori Export\ sort\ order=Ekspor urutan penyortiran @@ -1430,7 +1427,6 @@ Protected\ terms\ file=Istilah yang dilindungi Style\ file=File gaya Open\ OpenOffice/LibreOffice\ connection=Buka koneksi OpenOffice / LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Anda harus memasukkan setidaknya satu nama field Non-ASCII\ encoded\ character\ found=Karakter yang dikodekan non-ASCII ditemukan Toggle\ web\ search\ interface=Alihkan antarmuka pencarian web @@ -1551,7 +1547,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Menghapus semua Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Tidak dapat mengambil data entri dari ' % 0 '. Entry\ from\ %0\ could\ not\ be\ parsed.=Entri dari % 0 tidak dapat diuraikan. Invalid\ identifier\:\ '%0'.=Pengenal tidak valid\: ' % 0 '. -This\ paper\ has\ been\ withdrawn.=Makalah ini telah ditarik. @@ -1649,6 +1644,8 @@ Default\ pattern=Pola bawaan + + diff --git a/src/main/resources/l10n/JabRef_it.properties b/src/main/resources/l10n/JabRef_it.properties index 6f249bb1436..fc937b51fa7 100644 --- a/src/main/resources/l10n/JabRef_it.properties +++ b/src/main/resources/l10n/JabRef_it.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 voci +Export\ operation\ finished\ successfully.=L'operazione di esportazione si è conclusa con successo. + +Reveal\ in\ File\ Explorer=Mostra nel File Explorer %0\ matches\ the\ regular\ expression\ %1=%0 corrisponde all'espressione regolare %1 @@ -150,8 +153,6 @@ Cite\ command=Comando Cite Clear=Svuota -Clear\ fields=Annulla i campi - Open\ /\ close\ entry\ editor=Apri / chiudi editor voci Close\ dialog=Chiudi la finestra di dialogo @@ -700,11 +701,25 @@ Remove\ selected\ entries\ from\ this\ group=Rimuovi da questo gruppo le voci se Remove\ group=Rimuovi gruppo -Remove\ group\ "%0"?=Rimuovere il gruppo "%0"? +Remove\ group\ and\ subgroups=Rimuovi gruppo e sottogruppi + +Remove\ groups\ and\ subgroups=Rimuovi gruppi e sottogruppi + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Rimuovere tutti i gruppi selezionati e mantenere i loro sottogruppi? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Rimuovere il gruppo "%0" e mantenere i suoi sottogruppi? + +Remove\ groups=Rimuovere i gruppi + +Removed\ all\ selected\ groups.=Rimossi tutti i gruppi selezionati. Remove\ group\ "%0"\ and\ its\ subgroups?=Rimuovere il gruppo "%0" ed i suoi sottogruppi? -Remove\ group\ and\ subgroups=Rimuovi gruppo e sottogruppi +Removed\ group\ "%0"\ and\ its\ subgroups.=Rimosso gruppo "%0" e suoi sottogruppi. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Rimuovere tutti i gruppi selezionati e i loro sottogruppi? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Rimossi tutti i gruppi selezionati e i loro sottogruppi. Remove\ link=Rimuovere il collegamento @@ -714,8 +729,6 @@ Remove\ string\ %0=Rimuovi la stringa %0 Removed\ group\ "%0".=Rimosso gruppo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Rimosso gruppo "%0" e suoi sottogruppi. - Removed\ string=Stringa rimossa Renamed\ string=Stringa rinominata @@ -787,9 +800,6 @@ Select\ file\ from\ ZIP-archive=Seleziona un file da un archivio ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selezionare i nodi dell'albero per vedere ed accettare o rifiutare le modifiche -Set\ field=Imposta il campo -Set\ fields=Imposta i campi - Settings=Parametri Shortcut=Scorciatoia @@ -822,7 +832,12 @@ Size=Dimensione Skipped\ -\ No\ PDF\ linked=Saltato - Nessun file PDF collegato Skipped\ -\ PDF\ does\ not\ exist=Saltato - Il file PDF non esiste -Skipped\ entry.=Voce saltata +JabRef\ skipped\ the\ entry.=JabRef ha saltato la voce. +Import\ error=Errore di importazione +Open\ library\ error=Errore di apertura libreria +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Controlla il file della tua libreria per sintassi sbagliata. +SourceTab\ error=Errore SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=L'input dell'utente tramite entry-editor nella scheda `{}bibtex source` ha portato al fallimento. Sort\ subgroups=Ordina sottogruppi @@ -969,10 +984,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Riga %0\: Tro No\ full\ text\ document\ found=Nessun documento di testo completo trovato Download\ from\ URL=Scarica dall'URL Rename\ field=Rinomina il campo -Append\ field=Aggiungi campo -Append\ to\ fields=Aggiungi ai campi -Rename\ field\ to=Rinomina il campo in -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Sposta il contenuto di un campo in un campo con nome diverso Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Impossibile utilizzare la porta %0 per operazioni remote; la porta potrebbe essere in uso da parte di un'altra applicazione. Provare a specificare una porta diversa. @@ -1228,6 +1239,12 @@ Use\ custom\ proxy\ configuration=Utilizza una configurazione del proxy personal Proxy\ requires\ authentication=Il proxy richiede un'autenticazione Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Attenzione\: Password salvata come testo semplice\! Clear\ connection\ settings=Reinizializza i parametri di connessione +Check\ Proxy\ Setting=Controlla Impostazioni Proxy +Check\ connection=Controlla la connessione +Connection\ failed\!=Connessione fallita\! +Connection\ successful\!=Connessione riuscita\! + +SSL\ configuration\ changed=Configurazione SSL modificata Open\ folder=Apri cartella Export\ sort\ order=Esporta il modo di ordinamento @@ -1264,6 +1281,7 @@ Please\ open\ %0\ manually.=Per favore aprire %0 manualmente. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Il link è stato copiato negli appunti. Open\ %0\ file=Apri il file %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Impossibile rilevare il terminale automaticamente. Si prega di definire un terminale personalizzato nelle preferenze. Cannot\ delete\ file=Non posso cancellare il file File\ permission\ error=Errore nei permessi del file @@ -1640,6 +1658,8 @@ Issue\ report\ successful=Segnalazione del problema avvenuta con successo Your\ issue\ was\ reported\ in\ your\ browser.=Il problema è stato segnalato nel browser. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Il log e le informazioni dell'eccezione sono stati copiati negli appunti. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Per favore, incolla questa informazione (con Ctrl+V) nella descrizione del problema. +Last\ notification=Ultima notifica +Check\ the\ event\ log\ to\ see\ all\ notifications=Controlla il registro eventi per vedere tutte le notifiche Host=Host Port=Porta @@ -1677,7 +1697,6 @@ Protected\ terms\ file=File di termini protetti Style\ file=File stile Open\ OpenOffice/LibreOffice\ connection=Apri connessione OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Devi inserire almeno il nome di un campo Non-ASCII\ encoded\ character\ found=Trovato un carattere non codificato ASCII Non-UTF-8\ encoded\ field\ found=Trovato campo codificato non-UTF-8 Toggle\ web\ search\ interface=Inverti la selezione dell'interfaccia di ricerca web @@ -1815,7 +1834,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Rimuove tutte l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Impossibile recuperare dati in ingresso da '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Non è stato possibile analizzare la voce %0. Invalid\ identifier\:\ '%0'.=Identificatore non valido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Questo documento è stato ritirato. empty\ citation\ key=chiave BibTeX vuota Aux\ file=File aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Gruppo contenente voci citate in un determinato file TeX @@ -2298,10 +2316,6 @@ Removes\ digits.=Rimuove le cifre. Presets=Profili -Check\ Proxy\ Setting=Controlla Impostazioni Proxy -Check\ connection=Controlla la connessione -Connection\ failed\!=Connessione fallita\! -Connection\ successful\!=Connessione riuscita\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Genera gruppi da parole chiave nel campo seguente Generate\ groups\ for\ author\ last\ names=Genera gruppi in base al cognome dell'autore Regular\ expression=Espressione regolare @@ -2359,7 +2373,6 @@ Entry\ Type=Tipo di voce Entry\ types=Tipi di voce Field\ names=Nome del campo Others=Altri -Overwrite\ existing\ field\ values=Sovrascrivi i valori esistenti del campo Recommended=Consigliato Authors\ and\ Title=Autori e Titolo @@ -2427,3 +2440,23 @@ Search\ results\ from\ open\ libraries=Risultati della ricerca da librerie apert Custom\ API\ key=Chiave API personalizzata Check\ %0\ API\ Key\ Setting=Controlla le impostazioni della chiave API %0 + +Edit\ content=Modifica il contenuto +Copy\ or\ Move\ content=Copia o Sposta il contenuto +Overwrite\ field\ content=Sovrascrivi il contenuto del campo +Set=Imposta +Append=Accoda +Clear\ field\ content=Cancella il contenuto del campo +Set\ or\ append\ content=Imposta o aggiungi il contenuto +Edit\ field\ content\ for\ selected\ entries=Modifica il contenuto del campo per le voci selezionate +Rename=Rinomina +New\ field\ name=Nuovo nome del campo +Copy\ content=Copia il contenuto +Move\ content=Sposta il contenuto +Swap\ content=Scambia il contenuto +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copia o sposta il contenuto di un campo in un altro +Automatic\ field\ editor=Editor automatico dei campi + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Nota\: se le voci originali mancano di parole chiave per qualificarsi per la nuova configurazione di gruppo, confermando qui le aggiungeranno) +Assign=Assegna +Do\ not\ assign=Non assegnare diff --git a/src/main/resources/l10n/JabRef_ja.properties b/src/main/resources/l10n/JabRef_ja.properties index d250ae84767..580a657d479 100644 --- a/src/main/resources/l10n/JabRef_ja.properties +++ b/src/main/resources/l10n/JabRef_ja.properties @@ -20,6 +20,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 項目 + %0\ matches\ the\ regular\ expression\ %1=%0は正規表現%1に一致します %0\ matches\ the\ term\ %1=%0は項目%1に一致します @@ -150,8 +151,6 @@ Cite\ command=Citeコマンド Clear=消去 -Clear\ fields=フィールドを消去 - Open\ /\ close\ entry\ editor=項目エディタを開閉する Close\ dialog=ダイアログを閉じる @@ -695,11 +694,18 @@ Remove\ selected\ entries\ from\ this\ group=このグループから選択項 Remove\ group=グループを削除 -Remove\ group\ "%0"?=グループ「%0」を削除しますか? +Remove\ group\ and\ subgroups=グループと下層グループを削除 + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=グループ「%0」とその下層グループを削除しますか? -Remove\ group\ and\ subgroups=グループと下層グループを削除 +Removed\ group\ "%0"\ and\ its\ subgroups.=グループ「%0」とその下層グループを削除しました. + + Remove\ link=リンクを削除 @@ -709,8 +715,6 @@ Remove\ string\ %0=文字列 %0 を削除 Removed\ group\ "%0".=グループ「%0」を削除しました. -Removed\ group\ "%0"\ and\ its\ subgroups.=グループ「%0」とその下層グループを削除しました. - Removed\ string=文字列を削除しました Renamed\ string=文字列を改名しました @@ -779,9 +783,6 @@ Select\ file\ from\ ZIP-archive=ZIP書庫からファイルを選択してくだ Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=ツリーノードを選択して表示させ,変更を受諾ないし拒否してください -Set\ field=フィールドを設定 -Set\ fields=フィールドを設定 - Settings=設定 Shortcut=ショートカット @@ -814,7 +815,6 @@ Size=サイズ Skipped\ -\ No\ PDF\ linked=跳ばしました - PDFがリンクされていません Skipped\ -\ PDF\ does\ not\ exist=跳ばしました - PDFが存在しません -Skipped\ entry.=項目を跳ばしました. Sort\ subgroups=サブグループのソート @@ -960,10 +960,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=%0行目\: No\ full\ text\ document\ found=文書本体が見つかりません Download\ from\ URL=URLからダウンロード Rename\ field=フィールドを改名しました -Append\ field=フィールドを追加 -Append\ to\ fields=フィールドに追加 -Rename\ field\ to=フィールド名を以下に変更 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=フィールドの内容を別名のフィールドに移動する Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=リモート操作用に%0ポートを使用することができません.別のアプリケーションが使用している可能性があります.別のポートを指定してみてください. @@ -1219,6 +1215,11 @@ Use\ custom\ proxy\ configuration=ユーザー定義のプロキシ設定を用 Proxy\ requires\ authentication=プロキシには認証が必要 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=警告:パスワードは平文で保管されます! Clear\ connection\ settings=接続設定を消去する +Check\ Proxy\ Setting=プロキシ設定を確認 +Check\ connection=接続を確認 +Connection\ failed\!=接続に失敗しました! +Connection\ successful\!=接続に成功しました ! + Open\ folder=フォルダを開く Export\ sort\ order=書き出し整序順 @@ -1668,7 +1669,6 @@ Protected\ terms\ file=予約語ファイル Style\ file=スタイルファイル Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice接続を開く -You\ must\ enter\ at\ least\ one\ field\ name=少なくとも一つのフィールド名を入力する必要があります Non-ASCII\ encoded\ character\ found=ASCIIエンコードでない文字が検出されました Non-UTF-8\ encoded\ field\ found=UTF-8でエンコードされていないフィールドが見つかりました Toggle\ web\ search\ interface=ウェブ検索インタフェースを入切 @@ -1806,7 +1806,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=フィールド Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' から項目データを取得することができませんでした. Entry\ from\ %0\ could\ not\ be\ parsed.=%0 からの項目を解析することができませんでした. Invalid\ identifier\:\ '%0'.=「%0」は識別子として無効です. -This\ paper\ has\ been\ withdrawn.=この論文は撤回されました. empty\ citation\ key=空の文献キー Aux\ file=auxファイル Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=特定のTeXファイルでの引用項目を集めたグループ @@ -2141,10 +2140,6 @@ Removes\ digits.=数字を削除します。 Presets=プリセット -Check\ Proxy\ Setting=プロキシ設定を確認 -Check\ connection=接続を確認 -Connection\ failed\!=接続に失敗しました! -Connection\ successful\!=接続に成功しました ! Generate\ groups\ from\ keywords\ in\ the\ following\ field=次のフィールドのキーワードからグループを生成する Generate\ groups\ for\ author\ last\ names=著者の姓でグループを生成する Regular\ expression=正規表現 @@ -2190,7 +2185,6 @@ Entry\ Type=項目型 Entry\ types=項目型 Field\ names=フィールド名\: Others=その他 -Overwrite\ existing\ field\ values=既存のフィールド値を上書き Recommended=推奨 Authors\ and\ Title=著者とタイトル @@ -2247,3 +2241,5 @@ Search\ results\ from\ open\ libraries=オープンライブラリからの検 + + diff --git a/src/main/resources/l10n/JabRef_ko.properties b/src/main/resources/l10n/JabRef_ko.properties index 2938d9c870e..90f720b7e6f 100644 --- a/src/main/resources/l10n/JabRef_ko.properties +++ b/src/main/resources/l10n/JabRef_ko.properties @@ -20,6 +20,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 항목 + %0\ matches\ the\ regular\ expression\ %1=%0은 정규식 %1와 일치합니다. %0\ matches\ the\ term\ %1=%0은 항목 %1 와 일치합니다. @@ -150,8 +151,6 @@ Cite\ command=인용 명령 Clear=초기화 -Clear\ fields=필드 초기화 - Open\ /\ close\ entry\ editor=항목 편집기 열기/닫기 Close\ dialog=대화상자 닫기 @@ -693,11 +692,18 @@ Remove\ selected\ entries\ from\ this\ group=이 그룹에서 선택한 항목 Remove\ group=그룹 삭제 -Remove\ group\ "%0"?=그룹 "%0"을 삭제하겠습니까? +Remove\ group\ and\ subgroups=그룹 및 하위 그룹 제거 + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=그룹 "%0" 및 해당 하위 그룹을 제거하시겠습니까? -Remove\ group\ and\ subgroups=그룹 및 하위 그룹 제거 +Removed\ group\ "%0"\ and\ its\ subgroups.=그룹 "%0" 및 해당 하위 그룹을 제거했습니다. + + Remove\ link=링크 삭제 @@ -707,8 +713,6 @@ Remove\ string\ %0=문자열 %0 삭제 Removed\ group\ "%0".=그룹 %0 삭제 -Removed\ group\ "%0"\ and\ its\ subgroups.=그룹 "%0" 및 해당 하위 그룹을 제거했습니다. - Removed\ string=문자열 삭제 완료 Renamed\ string=문자열 이름 변경 완료 @@ -776,9 +780,6 @@ Select\ file\ from\ ZIP-archive=ZIP 아카이브에서 파일 선택 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=변경 사항을 보고 수락하거나 거부할 트리 노드를 선택합니다. -Set\ field=필드 설정 -Set\ fields=필드 설정 - Settings=설정 Shortcut=단축키 @@ -811,7 +812,6 @@ Size=크기 Skipped\ -\ No\ PDF\ linked=건너뜀 - 연결된 PDF 없음 Skipped\ -\ PDF\ does\ not\ exist=건너뜀 - PDF가 존재하지 않습니다 -Skipped\ entry.=항목을 건너뛰었습니다. Sort\ subgroups=하위 그룹 정렬 @@ -955,10 +955,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=%0행\: 손 No\ full\ text\ document\ found=전체 텍스트 문서를 찾을 수 없습니다. Download\ from\ URL=URL에서 다운로드 Rename\ field=필드 이름 변경 -Append\ field=필드 추가 -Append\ to\ fields=필드에 추가 -Rename\ field\ to=필드 이름 변경 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=필드 내용을 다른 이름의 필드로 이동 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=원격 작업에 포트 %0을 사용할 수 없습니다. 다른 응용 프로그램에서 사용 중일 수 있습니다. 다른 포트를 지정해 보십시오. @@ -1214,6 +1210,11 @@ Use\ custom\ proxy\ configuration=사용자 지정 프록시 구성 사용 Proxy\ requires\ authentication=프록시에 인증이 필요합니다. Attention\:\ Password\ is\ stored\ in\ plain\ text\!=주의\: 비밀번호는 일반 텍스트로 저장됩니다\! Clear\ connection\ settings=연결 설정 지우기 +Check\ Proxy\ Setting=프록시 설정 확인 +Check\ connection=연결 확인 +Connection\ failed\!=연결 실패\! +Connection\ successful\!=연결 성공\! + Open\ folder=폴더 열기 Export\ sort\ order=정렬 순서 내보내기 @@ -1663,7 +1664,6 @@ Protected\ terms\ file=보호된 용어 파일 Style\ file=파일 스타일 Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice 연결 열기 -You\ must\ enter\ at\ least\ one\ field\ name=하나 이상의 필드 이름을 입력해야 합니다. Non-ASCII\ encoded\ character\ found=ASCII가 아닌 인코딩된 문자를 찾았습니다. Non-UTF-8\ encoded\ field\ found=UTF-8이 아닌 인코딩 필드를 찾았습니다. Toggle\ web\ search\ interface=웹 검색 인터페이스 전환 @@ -1801,7 +1801,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=필드 내용 Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0'에서 항목 데이터를 검색할 수 없습니다. Entry\ from\ %0\ could\ not\ be\ parsed.=%0의 항목을 구문 분석할 수 없습니다. Invalid\ identifier\:\ '%0'.=잘못된 식별자\: '%0'. -This\ paper\ has\ been\ withdrawn.=이 논문은 철회되었습니다. empty\ citation\ key=빈 인용 키 Aux\ file=AUX 파일 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=주어진 TeX 파일에서 인용된 항목을 포함하는 그룹 @@ -1905,7 +1904,7 @@ Add\ new\ String=문자열 추가 Must\ not\ be\ empty\!=비워둘 수 없습니다\! Open\ Help\ page=도움말 열기 Add\ new\ field\ name=새 필드 이름 추가 -Field\ name\:=필드 이름\: +Field\ name\:=필드 이름\: Field\ name\ "%0"\ already\ exists=필드 이름 "%0"이 이미 존재합니다 No\ field\ name\ selected\!=필드 이름을 선택하지 않았습니다 Remove\ field\ name=필드 이름 제거 @@ -2280,10 +2279,6 @@ Removes\ digits.=숫자 제거 Presets=프리셋 -Check\ Proxy\ Setting=프록시 설정 확인 -Check\ connection=연결 확인 -Connection\ failed\!=연결 실패\! -Connection\ successful\!=연결 성공\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=다음 필드의 키워드에서 그룹 생성 Generate\ groups\ for\ author\ last\ names=작가의 성씨를 가지고 그룹 생성 Regular\ expression=정규식 @@ -2340,7 +2335,6 @@ Entry\ Type=목록 유형 Entry\ types=목록 유형들 Field\ names=필드 이름 Others=기타 -Overwrite\ existing\ field\ values=기존의 필드 값 덮어쓰기 Recommended=권장 Authors\ and\ Title=작가와 제목 @@ -2401,3 +2395,5 @@ Keep\ dialog\ always\ on\ top=대화창 위로 고정 + + diff --git a/src/main/resources/l10n/JabRef_nl.properties b/src/main/resources/l10n/JabRef_nl.properties index 6b183bbe7f9..312e3c7e358 100644 --- a/src/main/resources/l10n/JabRef_nl.properties +++ b/src/main/resources/l10n/JabRef_nl.properties @@ -1,3 +1,4 @@ +Could\ not\ delete\ empty\ entries.=Kan lege invoer niet verwijderen. @@ -13,6 +14,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 komt overeen met de standaard-uitdruk %1 %0\ matches\ the\ term\ %1=%0 komt overeen met de term %1 @@ -135,8 +137,6 @@ Cite\ command=Citeer opdracht Clear=Wissen -Clear\ fields=Velden wissen - Close\ dialog=Sluit dialoog @@ -638,11 +638,18 @@ Remove\ selected\ entries\ from\ this\ group=Geselecteerde items uit deze groep Remove\ group=Verwijder groep -Remove\ group\ "%0"?=Verwijder groep "%0"? +Remove\ group\ and\ subgroups=Verwijder groep en subgroepen + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Verwijder groep "%0" en zijn subgroepen? -Remove\ group\ and\ subgroups=Verwijder groep en subgroepen +Removed\ group\ "%0"\ and\ its\ subgroups.=Groep "%0" en zijn subgroepen verwijderd. + + Remove\ link=Verwijder link @@ -651,8 +658,6 @@ Remove\ old\ entry=Verwijder oude invoer Removed\ group\ "%0".=Groep "%0" verwijderd. -Removed\ group\ "%0"\ and\ its\ subgroups.=Groep "%0" en zijn subgroepen verwijderd. - Removed\ string=Tekenreeks verwijderd Renamed\ string=Tekenreeks hernoemd @@ -719,9 +724,6 @@ Select\ file\ from\ ZIP-archive=Selecteer bestand van ZIP-archief Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecteer de boom knopen om veranderingen te tonen en te accepteren of afwijzen -Set\ field=Veld instellen -Set\ fields=Velden instellen - Settings=Instellingen Shortcut=Snelkoppeling @@ -754,7 +756,6 @@ Size=Grootte Skipped\ -\ No\ PDF\ linked=Overgeslagen - Geen PDF gelinkt Skipped\ -\ PDF\ does\ not\ exist=Overgeslagen - PDF bestaat niet -Skipped\ entry.=Overgeslagen invoer. source\ edit=broncode aanpassen @@ -872,10 +873,6 @@ MIME\ type=MIME type No\ full\ text\ document\ found=Geen volledig tekstdocument gevonden Download\ from\ URL=Download vanaf URL Rename\ field=Veld hernoemen -Append\ field=Veld toevoegen -Append\ to\ fields=Toevoegen aan velden -Rename\ field\ to=Veld hernoemen naar -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Verplaats de inhoud van een veld naar een veld met een andere naam Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan poort %0 niet gebruiken voor bediening op afstand; een andere applicatie heeft deze mogelijk in gebruik. Probeer een andere poort op te geven. @@ -1090,6 +1087,7 @@ Proxy\ requires\ authentication=Proxy vereist authenticatie Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Opgelet\: Wachtwoord wordt opgeslagen in platte tekst\! Clear\ connection\ settings=Verbindingsinstellingen wissen + Open\ folder=Map openen Export\ sort\ order=Sorteervolgorde exporteren @@ -1494,7 +1492,6 @@ Protected\ terms\ file=Beschermde voorwaarden bestand Style\ file=Bestandsstijl Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice verbinding openen -You\ must\ enter\ at\ least\ one\ field\ name=U moet ten minste één veldnaam opgeven Non-ASCII\ encoded\ character\ found=Niet-ASCII gecodeerd teken gevonden Toggle\ web\ search\ interface=Web zoek interface in-/uitschakelen @@ -1615,7 +1612,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Verwijdert alle Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Kon geen data ophalen van '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Invoer van %0 kon niet worden geparseerd. Invalid\ identifier\:\ '%0'.=Ongeldig Id\: '%0'. -This\ paper\ has\ been\ withdrawn.=Dit artikel is ingetrokken. Aux\ file=Aux bestand Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Groep bevat invoergegevens aangehaald in een bepaald TeX-bestand @@ -1763,6 +1759,8 @@ Default\ pattern=Standaard patroon + + diff --git a/src/main/resources/l10n/JabRef_no.properties b/src/main/resources/l10n/JabRef_no.properties index 164dcbe8bb2..d8de347be33 100644 --- a/src/main/resources/l10n/JabRef_no.properties +++ b/src/main/resources/l10n/JabRef_no.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 matcher regulæruttrykket %1 %0\ matches\ the\ term\ %1=%0 matcher uttrykket %1 @@ -126,8 +127,6 @@ Cite\ command=Siteringskommando Clear=Opphev -Clear\ fields=Slett felter - Close\ dialog=Lukk dialog @@ -604,11 +603,18 @@ Remove\ selected\ entries\ from\ this\ group=Fjern valgte oppføringer fra denne Remove\ group=Fjern gruppe -Remove\ group\ "%0"?=Fjern gruppen "%0"? +Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Fjern gruppen "%0" og dens undergrupper? -Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernet gruppen "%0" og dens undergrupper + + Remove\ link=Slett link @@ -617,8 +623,6 @@ Remove\ old\ entry=Fjern gammel enhet Removed\ group\ "%0".=Fjernet gruppen "%0" -Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernet gruppen "%0" og dens undergrupper - Removed\ string=Fjernet streng Renamed\ string=Endret navn på streng @@ -681,9 +685,6 @@ Select\ file\ from\ ZIP-archive=Velg fil fra ZIP-fil Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Velg trenodene for å inspisere og akseptere eller avslå endringer -Set\ field=Sett felt -Set\ fields=Sett felter - Settings=Innstillinger Shortcut=Snarvei @@ -714,7 +715,6 @@ Size=Størrelse Skipped\ -\ No\ PDF\ linked=Hoppet over - ingen PDF-fil linket Skipped\ -\ PDF\ does\ not\ exist=Hoppet over - PDF-filen finnes ikke -Skipped\ entry.=Hoppet over enhet. source\ edit=redigering av kilde @@ -828,8 +828,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Last ned fra URL Rename\ field=Endre navn på felt -Rename\ field\ to=Endre navn på felt til -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytt innholdet i et felt til et annet felt Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan ikke bruke port %0 for fjernstyring; den kan være i bruk av et annet program. Pr¸v å spesifisere en annen port. @@ -962,6 +960,7 @@ Finished\ automatically\ setting\ external\ links.=Fullførte automatisk setting + Open\ folder=Åpne mapp Export\ sort\ order=Sorteringsrekkefølge ved eksport @@ -1203,6 +1202,8 @@ Default\ pattern=Standardmønster + + diff --git a/src/main/resources/l10n/JabRef_pl.properties b/src/main/resources/l10n/JabRef_pl.properties index 153fa9456a9..e8a82418acc 100644 --- a/src/main/resources/l10n/JabRef_pl.properties +++ b/src/main/resources/l10n/JabRef_pl.properties @@ -13,6 +13,7 @@ + Abbreviation=Skrót About\ JabRef=O JabRef @@ -95,8 +96,6 @@ Change\ entry\ type=Zmień typ wpisu Clear=Wyczyść -Clear\ fields=Wyczyść pola - Close\ dialog=Zamknij okno @@ -472,9 +471,16 @@ Remove\ subgroups=Usuń podgrupy Remove\ group=Usuń grupę +Remove\ group\ and\ subgroups=Usuń grupę i podgrupy + + + + + + + -Remove\ group\ and\ subgroups=Usuń grupę i podgrupy Remove\ link=Usuń łącze @@ -484,7 +490,6 @@ Remove\ old\ entry=Usuń stary wpis - Replace=Zamień Replace\ With\:=Zamień na\: Find\:=Znajdź\: @@ -529,7 +534,6 @@ Select\ entry\ type=Wybierz typ wpisu - Settings=Ustawienia Shortcut=Skrót @@ -551,7 +555,6 @@ Simple\ HTML=Uproszczony HTML Size=Rozmiar -Skipped\ entry.=Pominięty wpis. @@ -627,7 +630,6 @@ MIME\ type=Typ MIME Download\ from\ URL=Pobierz z adresu URL Rename\ field=Zmień nazwę pola -Append\ to\ fields=Dołącz do pól Please\ enter\ a\ valid\ file\ path.=Wprowadź prawidłową ścieżkę do pliku. @@ -717,6 +719,7 @@ Hostname=Nazwa hosta Proxy\ configuration=Konfiguracja proxy + Open\ folder=Otwórz folder Parsing\ error=Błąd parsowania @@ -1077,6 +1080,8 @@ Reset=Reset + + diff --git a/src/main/resources/l10n/JabRef_pt.properties b/src/main/resources/l10n/JabRef_pt.properties index a4cf86dab14..6fd581ffae8 100644 --- a/src/main/resources/l10n/JabRef_pt.properties +++ b/src/main/resources/l10n/JabRef_pt.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 encontrada a expressão regular %1 %0\ matches\ the\ term\ %1=%0 encontrado o termo %1 @@ -130,8 +131,6 @@ Cite\ command=Comando citar Clear=Limpar -Clear\ fields=Limpar campos - Close\ dialog=Fechar janela de diálogo @@ -629,11 +628,18 @@ Remove\ entry\ from\ import=Remover referência da importação Remove\ group=Remover grupo -Remove\ group\ "%0"?=Remover grupo "%0"? +Remove\ group\ and\ subgroups=Remover grupos e subgrupos + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Remover grupo "%0" e seus subgrupos? -Remove\ group\ and\ subgroups=Remover grupos e subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. + + Remove\ link=Remover link @@ -642,8 +648,6 @@ Remove\ old\ entry=Remover referência antiga Removed\ group\ "%0".=Grupo "%0" removido. -Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. - Removed\ string=String removida Renamed\ string=String renomeada @@ -696,9 +700,6 @@ Select\ file\ from\ ZIP-archive=Selecionar arquivo a partir de um arquivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecione os nós da árvore para visualizar e aceitar ou rejeitar mudanças -Set\ field=Configurar campo -Set\ fields=Configurar campos - Settings=Configurações Shortcut=Atalho @@ -729,7 +730,6 @@ Size=Tamanho Skipped\ -\ No\ PDF\ linked=Omitido - Nenhum PDF linkado Skipped\ -\ PDF\ does\ not\ exist=Omitido - O PDF não existe -Skipped\ entry.=Referência omitida. source\ edit=edição de fonte @@ -840,8 +840,6 @@ Unable\ to\ open\ link.=Não foi possível abrir link. This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ running\ instance\ of\ JabRef\ instead\ of\ opening\ a\ new\ instance.\ For\ instance,\ this\ is\ useful\ when\ you\ open\ a\ file\ in\ JabRef\ from\ your\ web\ browser.\ Note\ that\ this\ will\ prevent\ you\ from\ running\ more\ than\ one\ instance\ of\ JabRef\ at\ a\ time.=Esta funcionalidade permite que novos arquivos sejam abertos ou importados para uma instância do JabRef já aberta ao invés de abrir uma nova instância. Por exemplo, isto é útil quando você abre um arquivo no JabRef a partir de ser navegador web. Note que isto irá previnir que você execute uma ou mais instâncias do JabRef ao mesmo tempo. Rename\ field=Renomear campo -Rename\ field\ to=Renomear campo para -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover conteúdo de um campo para um campo com nome diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Não é possível utilizar a porta %0 para operação remota; outra aplicação pode estar usando-a. Tente utilizar uma outra porta. @@ -1050,6 +1048,7 @@ Use\ custom\ proxy\ configuration=Usar configurações personalizadas de proxy Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atenção\: A senha é armazenada em texto simples\! Clear\ connection\ settings=Limpar configurações da conexão + Open\ folder=Abrir diretório Export\ sort\ order=Exportar ordenação @@ -1355,6 +1354,8 @@ Default\ pattern=Ppadrão predefinido + + diff --git a/src/main/resources/l10n/JabRef_pt_BR.properties b/src/main/resources/l10n/JabRef_pt_BR.properties index f75024aafc7..179f7dd8128 100644 --- a/src/main/resources/l10n/JabRef_pt_BR.properties +++ b/src/main/resources/l10n/JabRef_pt_BR.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entradas -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 exportação bem-sucedida. Abrir a pasta que contém o arquivo salvo? + %0\ matches\ the\ regular\ expression\ %1=%0 encontrada a expressão regular %1 @@ -151,8 +151,6 @@ Cite\ command=Comando citar Clear=Limpar -Clear\ fields=Limpar campos - Open\ /\ close\ entry\ editor=Abrir / fechar o editor de referências Close\ dialog=Fechar janela de diálogo @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Remover as referências selecionada Remove\ group=Remover grupo -Remove\ group\ "%0"?=Remover grupo "%0"? +Remove\ group\ and\ subgroups=Remover grupos e subgrupos + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Remover grupo "%0" e seus subgrupos? -Remove\ group\ and\ subgroups=Remover grupos e subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. + + Remove\ link=Remover link @@ -717,8 +722,6 @@ Remove\ string\ %0=Remover string %0 Removed\ group\ "%0".=Grupo "%0" removido. -Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. - Removed\ string=String removida Renamed\ string=String renomeada @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=Selecionar arquivo a partir de um arquivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecione os nós da árvore para visualizar e aceitar ou rejeitar mudanças -Set\ field=Configurar campo -Set\ fields=Configurar campos - Settings=Configurações Shortcut=Atalho @@ -825,7 +825,6 @@ Size=Tamanho Skipped\ -\ No\ PDF\ linked=Omitido - Nenhum PDF linkado Skipped\ -\ PDF\ does\ not\ exist=Omitido - O PDF não existe -Skipped\ entry.=Referência omitida. Sort\ subgroups=Ordenar subgrupos @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Linha %0\: En No\ full\ text\ document\ found=Nenhum documento encontrado Download\ from\ URL=Baixar da URL Rename\ field=Renomear campo -Append\ field=Acrescentar campo -Append\ to\ fields=Acrescentar aos campos -Rename\ field\ to=Renomear campo para -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover conteúdo de um campo para um campo com nome diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Não é possível utilizar a porta %0 para operação remota; outra aplicação pode estar usando-a. Tente utilizar uma outra porta. @@ -1242,6 +1237,11 @@ Use\ custom\ proxy\ configuration=Usar configurações personalizadas de proxy Proxy\ requires\ authentication=Proxy requer autenticação Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atenção\: A senha é armazenada em texto simples\! Clear\ connection\ settings=Limpar configurações da conexão +Check\ Proxy\ Setting=Verificar Configuração de Proxy +Check\ connection=Verificar conexão +Connection\ failed\!=Falha na conexão\! +Connection\ successful\!=Conexão bem-sucedida\! + Open\ folder=Abrir diretório Export\ sort\ order=Exportar ordenação @@ -1694,7 +1694,6 @@ Protected\ terms\ file=Arquivo de termos protegidos Style\ file=Arquivo de estilo Open\ OpenOffice/LibreOffice\ connection=Abrir conexão com OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Você deve digitar pelo menos um nome de campo Non-ASCII\ encoded\ character\ found=Caractere não-ASCII encontrado Non-UTF-8\ encoded\ field\ found=Campo não-UTF-8 encontrado Toggle\ web\ search\ interface=Alternar interface de busca na web @@ -1832,7 +1831,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Remove todas as Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Não foi possível recuperar dados de referência a partir de '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Não foi possível analisar a referência a partir de %0. Invalid\ identifier\:\ '%0'.=Identificador inválido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Este documento foi retirado. empty\ citation\ key=chave de citação vazia Aux\ file=Arquivo Aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo contendo referências citadas em um arquivo TeX @@ -2314,10 +2312,6 @@ Removes\ digits.=Remove dígitos. Presets=Predefinições -Check\ Proxy\ Setting=Verificar Configuração de Proxy -Check\ connection=Verificar conexão -Connection\ failed\!=Falha na conexão\! -Connection\ successful\!=Conexão bem-sucedida\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Gerar grupos de palavras-chave no campo seguinte Generate\ groups\ for\ author\ last\ names=Gerar grupos para os últimos nomes dos autores Regular\ expression=Expressão regular @@ -2375,7 +2369,6 @@ Entry\ Type=Tipo de referência Entry\ types=Tipos de referência Field\ names=Nomes dos campos Others=Outros -Overwrite\ existing\ field\ values=Sobrescrever valores de campos existentes Recommended=Recomendado Authors\ and\ Title=Autores e Título @@ -2439,3 +2432,5 @@ Search\ results\ from\ open\ libraries=Resultados da pesquisa para as biblioteca + + diff --git a/src/main/resources/l10n/JabRef_ru.properties b/src/main/resources/l10n/JabRef_ru.properties index 4f2f56445c7..0999c1ddcb0 100644 --- a/src/main/resources/l10n/JabRef_ru.properties +++ b/src/main/resources/l10n/JabRef_ru.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 записей +Export\ operation\ finished\ successfully.=Операция экспорта выполнена успешно. + +Reveal\ in\ File\ Explorer=Открыть в Проводнике %0\ matches\ the\ regular\ expression\ %1=%0 соответствует регулярному выражению %1 @@ -150,8 +153,6 @@ Cite\ command=Команда цитирования Clear=Очистить -Clear\ fields=Очистить поля - Open\ /\ close\ entry\ editor=Открыть / закрыть редактор записей Close\ dialog=Закрыть диалоговое окно @@ -173,6 +174,8 @@ Copy=Копировать Copy\ title=Копировать заглавие Copy\ \\cite{citation\ key}=Копировать \\ цитировать{citation key} +Copy\ citation\ (html)=Копировать цитирование (html) +Copy\ citation\ (text)=Копировать цитирование (текст) Copy\ citation\ key=Копировать ключ цитаты Copy\ citation\ key\ and\ link=Скопировать ключ цитаты и ссылку Copy\ citation\ key\ and\ title=Копировать ключ цитаты и заголовок @@ -230,6 +233,8 @@ Library\ properties=Свойства БД Default=По умолчанию +Character\ encoding\ UTF-8\ is\ not\ supported.=Кодировка UTF-8 не поддерживается. +UTF-8\ could\ not\ be\ used\ to\ encode\ the\ following\ characters\:\ %0=Не удалось использовать UTF-8 для кодирования следующих символов\: %0 The\ chosen\ encoding\ '%0'\ could\ not\ encode\ the\ following\ characters\:=В выбранной кодировке '%0' невозможно прочитать следующие символы\: Downloading=Загрузка … @@ -278,6 +283,7 @@ Downloaded\ website\ as\ an\ HTML\ file.=Загружен сайт в форма duplicate\ removal=удаление дубликатов +Duplicate\ fields=Дубликаты полей Duplicate\ string\ name=Дубликат имени строки @@ -536,7 +542,9 @@ Moved\ group\ "%0".=Перемещенная группа "%0". Mr.\ DLib\ Privacy\ settings=Настройки конфиденциальности сервиса "Mr. DLib" +No\ database\ is\ open=База данных не открыта +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=Нам требуется база данных для экспорта. Откройте её. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=Веб сервис "Mr. DLib" не предлагает рекомендаций для этой записи. @@ -695,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Удалить выбранные Remove\ group=Удалить группу -Remove\ group\ "%0"?=Удалить группу "%0"? +Remove\ group\ and\ subgroups=Удалить группу и подгруппы + +Remove\ groups\ and\ subgroups=Удалить группы и подгруппы + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Удалить все выбранные группы, сохранив их подгруппы? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Удалить группу "%0", сохранив её подгруппы? + +Remove\ groups=Удалить группы + +Removed\ all\ selected\ groups.=Все выбранные группы удалены. Remove\ group\ "%0"\ and\ its\ subgroups?=Удалить группу "%0" и ее подгруппы? -Remove\ group\ and\ subgroups=Удалить группу и подгруппы +Removed\ group\ "%0"\ and\ its\ subgroups.=Группа "%0" и ее подгруппы удалены. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Удалить все выбранные группы и их подгруппы? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Все выбранные группы и их подгруппы удалены. Remove\ link=Удалить ссылку @@ -709,8 +731,6 @@ Remove\ string\ %0=Удалить строку %0 Removed\ group\ "%0".=Группа "%0" удалена. -Removed\ group\ "%0"\ and\ its\ subgroups.=Группа "%0" и ее подгруппы удалены. - Removed\ string=Строка удалена Renamed\ string=Строка переименована @@ -733,6 +753,8 @@ Replaces\ Unicode\ ligatures\ with\ their\ expanded\ form=Заменяет ли Required\ fields=Обязательные поля +Do\ not\ resolve\ BibTeX\ strings=Не разрешать строки BibTeX +Resolve\ BibTeX\ strings\ for\ the\ following\ fields=Разрешить строки BibTeX для следующих полей resolved=разрешено @@ -740,11 +762,13 @@ Restart=Перезапустить Restart\ required=Требуется перезапуск +Return\ to\ dialog=Вернуться в диалоговое окно Review=Просмотр Review\ changes=Просмотр изменений Review\ Field\ Migration=Перемещение поля обзора +Loading=Загрузка Save=Сохранить Save\ all\ finished.=Сохранить все завершенные. @@ -778,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Выбрать файл из ZIP-архива Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Выбрать узлы дерева для просмотра и применения/отклонения изменений -Set\ field=Настройка поля -Set\ fields=Настройка полей - Settings=Параметры Shortcut=Ярлык @@ -813,7 +834,12 @@ Size=Размер Skipped\ -\ No\ PDF\ linked=Пропущено - Без ссылок на PDF Skipped\ -\ PDF\ does\ not\ exist=Пропущено - PDF не существует -Skipped\ entry.=Запись пропущена. +JabRef\ skipped\ the\ entry.=JabRef пропустил запись. +Import\ error=Ошибка импортирования +Open\ library\ error=Ошибка при открытии библиотеки +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Пожалуйста, проверьте ваш файл библиотеки на правильность синтаксиса. +SourceTab\ error=Ошибка SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Пользовательский ввод посредством редактора записей во вкладке `{}bibtex source` привёл к возникновению ошибки. Sort\ subgroups=Сортировать подгруппы @@ -902,6 +928,7 @@ Warning=Предупреждение Warnings=Предупреждения +Warning\:\ You\ added\ field\ "%0"\ twice.\ Only\ one\ will\ be\ kept.=Внимание\: Вы добавили поле "%0" дважды. Только одно из них будет сохранено. web\ link=веб-ссылки @@ -959,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Строка No\ full\ text\ document\ found=Не найдено полнотекстовых документов Download\ from\ URL=Загрузить из URL-адреса Rename\ field=Переименовать поле -Append\ field=Добавить поле -Append\ to\ fields=Добавить к полям -Rename\ field\ to=Переименовать поле в -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Переместить содержимое поля в поле с другим именем Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Невозможно использовать порт %0 для удаленного подключения; another application may be using it. Try specifying another port. @@ -1032,6 +1055,7 @@ Cite\ special=Особое цитирование Extra\ information\ (e.g.\ page\ number)=Дополнительные сведения (напр., номер страницы) Manage\ citations=Управление цитированием Problem\ modifying\ citation=Ошибка при модификации цитаты +Problem\ collecting\ citations=Возникла проблема при сборе ссылок Citation=Цитата Connecting...=Подключение... Select\ style=Выбор стиля @@ -1046,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=Синхронизация библи Select\ which\ open\ Writer\ document\ to\ work\ on=Выберите открытый документ OpenOffice/LibreOffice Writer для работы Connected\ to\ document=Подключено к документу +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Не удалось выполнить подключение к документу Writer. Убедитесь, что документ открыт перед использованием кнопки 'Выбрать документ Writer', чтобы выполнить подключение. +No\ Writer\ documents\ found=Документы Writer не найдены Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=Вставка цитаты без текста (запись отображается в списке ссылок) Cite\ selected\ entries\ with\ extra\ information=Цитировать выбранные записи с дополнительными сведениями @@ -1054,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=Проверьте актуал Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=Ваш документ OpenOffice/LibreOffice ссылается на ключ цитирования '%0', который не найден в вашей текущей библиотеке. +This\ operation\ requires\ a\ bibliography\ database.=Этой операции требуется библиографическая база данных. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Ссылки в вашем документе OpenOffice/LibreOffice содержат не менее %0 ключей цитирования, которые не удалось найти в текущей библиотеке. Некоторые из них %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=В вашем документе OpenOffice/LibreOffice отсутствуют какие-либо ключи цитирования, которые также можно найти в вашей текущей библиотеке. Unable\ to\ synchronize\ bibliography=Не удалось выполнить синхронизацию библиографии Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=Объединить пары цитат, разделенные только пробелом @@ -1063,7 +1092,12 @@ Autodetection\ failed=Ошибка автоопределения Please\ wait...=Подождите... Connection\ lost=Соединение прервано +Could\ not\ update\ bibliography=Не удалось обновить библиографию +Not\ connected\ to\ document=Нет подключения к документу +Problem\ combining\ cite\ markers=Проблема при объединении отметок цитирования +Unable\ to\ reload\ style\ file=Не удалось перезагрузить файл оформления +Problem\ during\ separating\ cite\ markers=Проблема при разделении отметок цитирования Automatically\ sync\ bibliography\ when\ inserting\ citations=Автоматически синхронизировать библиографию при вставке цитат Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=Поиск записей BibTeX только в активной вкладке @@ -1083,6 +1117,7 @@ Style\ selection=Выбор стиля No\ valid\ style\ file\ defined=Не определен допустимый файл стиля Choose\ pattern=Выбор шаблона Search\ and\ store\ files\ relative\ to\ library\ file\ location=Поиск и хранение файлов относительно файла библиотеки +File\ directory=Каталог файлов Could\ not\ run\ the\ gnuclient/emacsclient\ program.\ Make\ sure\ you\ have\ the\ emacsclient/gnuclient\ program\ installed\ and\ available\ in\ the\ PATH.=Не удалось запустить приложение gnuclient/emacsclient. Убедитесь, что приложение gnuclient/emacsclient установлено и доступно по указанному пути PATH. You\ must\ select\ either\ a\ valid\ style\ file,\ or\ use\ one\ of\ the\ default\ styles.=Необходимо использовать допустимый файл стиля или один из стилей по умолчанию. @@ -1165,6 +1200,7 @@ Toggle\ relevance=Показать/скрыть релевантность Toggle\ quality\ assured=Прказать/скрыть проверку контроля качества Toggle\ print\ status=Изменить статус 'печать' Update\ keywords=Изменить ключевые слова +Problem\ connecting=Проблема при подключении Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=Соединение с OpenOffice/LibreOffice прервано. Убедитесь, что приложение OpenOffice/LibreOffice запущено и повторите соединение. JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef отправляет по крайней мере один запрос для заметки издателю. @@ -1179,6 +1215,7 @@ Removed\ all\ subgroups\ of\ group\ "%0".=Все подгруппы группы To\ disable\ the\ memory\ stick\ mode\ rename\ or\ remove\ the\ jabref.xml\ file\ in\ the\ same\ folder\ as\ JabRef.=Для отключения режима флеш-памяти переименуйте или удалите файл jabref.xml в каталоге JabRef. Unable\ to\ connect.\ One\ possible\ reason\ is\ that\ JabRef\ and\ OpenOffice/LibreOffice\ are\ not\ both\ running\ in\ either\ 32\ bit\ mode\ or\ 64\ bit\ mode.=Не удалось выполнить подключение. Возможная причина\: JabRef и OpenOffice/LibreOffice не запущены в режиме 32-х или 64-х битного приложения. Delimiter(s)=Разделитель(и) +When\ downloading\ files,\ or\ moving\ linked\ files\ to\ the\ file\ directory,\ use\ the\ bib\ file\ location.=При загрузке файлов или перемещении связанных файлов в каталог файлов используйте расположение файла bib. Searching...=Выполняется поиск... Please\ enter\ a\ search\ string=Введите строку для поиска @@ -1216,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Использовать пользовате Proxy\ requires\ authentication=Для прокси-сервера требуется авторизация Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Внимание. Пароль сохранен в виде простого текста. Clear\ connection\ settings=Удалить настройки подключения +Check\ Proxy\ Setting=Проверить настройки прокси +Check\ connection=Проверить подключение +Connection\ failed\!=Соединение потеряно\! +Connection\ successful\!=Соединение успешно\! + +SSL\ Configuration=Конфигурация SSL +SSL\ configuration\ changed=Конфигурация SSL изменена +SSL\ certificate\ file=Файл сертификата SSL +Duplicate\ Certificates=Дубликаты сертификатов +You\ already\ added\ this\ certificate=Вы уже добавили этот сертификат Open\ folder=Открыть папку Export\ sort\ order=Экспорт в порядке сортировки @@ -1252,6 +1299,7 @@ Please\ open\ %0\ manually.=Откройте %0 вручную. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Ссылка скопирована в буфер обмена. Open\ %0\ file=Открыть файл %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Не удалось обнаружить терминал автоматически. Пожалуйста, укажите пользовательский терминал в настройках. Cannot\ delete\ file=Невозможно удалить файл File\ permission\ error=Ошибка прав доступа для файла @@ -1558,10 +1606,21 @@ Custom=Определено пользователем Export\ cited=Экспорт цитированного Unable\ to\ generate\ new\ library=Не удалось создать новую БД +The\ cursor\ is\ in\ a\ protected\ area.=Курсор находится в защищённой области. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=Не открыто библиографических баз данных для цитирования. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=Не открыто библиографических баз данных для обновления отметок ссылок после цитирования. +No\ bibliography\ entries\ are\ selected\ for\ citation.=Не выбраны библиографические записи для выполнения цитирования. +No\ bibliography\ style\ is\ selected\ for\ citation.=Не выбрано оформление библиографии для выполнения цитирования. +No\ database=Нет базы данных +No\ entries\ selected\ for\ citation=Не выбрано записей для выполнения цитирования +Open\ one\ before\ citing.=Откройте одну из них перед цитированием. +Select\ one\ before\ citing.=Выберите одну из них перед цитированием. +Select\ some\ before\ citing.=Выберите несколько перед выполнением цитирования. Found\ identical\ ranges=Найдены идентичные диапазоны Found\ overlapping\ ranges=Найдены пересекающиеся диапазоны @@ -1628,6 +1687,8 @@ Issue\ report\ successful=Отчёт об ошибке успешно отпра Your\ issue\ was\ reported\ in\ your\ browser.=Отчет о проблеме был отправлен в браузер. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Лог и информация об ошибке успешно скопированы. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Пожалуйста, вставьте эту информацию (с помощью Ctrl+V) в описание проблемы. +Last\ notification=Последнее уведомление +Check\ the\ event\ log\ to\ see\ all\ notifications=Проверьте журнал событий для просмотра всех уведомлений Host=Хост Port=Порт @@ -1665,7 +1726,6 @@ Protected\ terms\ file=Файл с защищенными терминами Style\ file=Файл стиля Open\ OpenOffice/LibreOffice\ connection=Установить подключение к Open OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Необходимо ввести минимум одно имя поля Non-ASCII\ encoded\ character\ found=Обнаружен символ не в кодировке ASCII Non-UTF-8\ encoded\ field\ found=Найдено поле НЕ в кодировке UTF-8 Toggle\ web\ search\ interface=Переключение интерфейса веб-поиска @@ -1803,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Удаляет Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Не удалось получить данные из записи '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Данные в %0 не могут быть обработаны. Invalid\ identifier\:\ '%0'.=Неверный идентификатор\: '%0'. -This\ paper\ has\ been\ withdrawn.=Настоящий документ был отозван. empty\ citation\ key=пустой ключ цитаты Aux\ file=Aux файл Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Группа, содержащая записи в заданном TeX файле @@ -1831,6 +1890,7 @@ Could\ not\ connect\ to\ Vim\ server.\ Make\ sure\ that\ Vim\ is\ running\ with\ Could\ not\ connect\ to\ a\ running\ gnuserv\ process.\ Make\ sure\ that\ Emacs\ or\ XEmacs\ is\ running,\ and\ that\ the\ server\ has\ been\ started\ (by\ running\ the\ command\ 'server-start'/'gnuserv-start').=Не удалось подключиться к запущенному процессу gnuserv. Убедитесь, что приложение Emacs или XEmacsи сервер запущены(с помощью команды 'server-start'/'gnuserv-start'). Error\ pushing\ entries=Ошибка при отправке записей +Preamble=Преамбула Markings=Маркировка Use\ selected\ instance=Использовать выбранный экземпляр @@ -1849,6 +1909,7 @@ Blog=Блог Check\ integrity=Проверка целостности Cleanup\ URL\ link=Очистить URL ссылку Cleanup\ URL\ link\ by\ removing\ special\ symbols\ and\ extracting\ simple\ link=Очистка URL ссылку, удалив специальные символы, чтобы получился простой URL +Copy\ DOI=Копировать DOI Copy\ DOI\ url=Копировать URL-адрес DOI Development\ version=Версия для разработчиков Export\ selected\ entries=Экспорт выбранных записей @@ -1858,6 +1919,8 @@ JabRef\ resources=Ресурсы JabRef Manage\ journal\ abbreviations=Настроить аббревиатуры журналов Manage\ protected\ terms=Управление защищенными терминами New\ entry\ from\ plain\ text=Новая запись из обычного текста +Import\ by\ ID=Импортировать по ID +Enter\ a\ valid\ ID=Введите действительный ID New\ sublibrary\ based\ on\ AUX\ file=Новая под-библиотека из файла AUX Push\ entries\ to\ external\ application\ (%0)=Передать записи во внешнее приложение (%0) Quit=Выход @@ -1917,6 +1980,7 @@ Keyword\ separator=Разделитель ключевых слов Remove\ keyword=Удалить слово Are\ you\ sure\ you\ want\ to\ remove\ keyword\:\ "%0"?=Вы уверены, что вы хотите удалить слово\: "%0"? Reset\ to\ default=Сбросить настройки +String\ constants=Строковые постоянные Export\ all\ entries=Экспорт всех записей Generate\ citation\ keys=Генерировать ключи цитирования Manage\ field\ names\ &\ content=Управлять названиями полей и контентом @@ -2046,6 +2110,7 @@ Please\ provide\ a\ valid\ aux\ file.=Пожалуйста, предоставь Keyword\ delimiter=Разделитель ключевых слов Hierarchical\ keyword\ delimiter=Разделитель иерархических ключевых слов Escape\ ampersands=Экранировка амперсандов +Escape\ dollar\ sign=Экранированный символ доллара Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Подсказка\:\nдля поиска всех полей со значением Smith, введите\:\nsmith\nДля поиска в поле Автор значения Smith а в поле Заголовок значения electrical, введите\:\nauthor\=smith and title\=electrical @@ -2234,7 +2299,7 @@ This\ entry\ type\ is\ intended\ for\ sources\ such\ as\ web\ sites\ which\ are\ A\ single-volume\ work\ of\ reference\ such\ as\ an\ encyclopedia\ or\ a\ dictionary.=Неделимая работа или ссылка, как энциклопедия или словарь. A\ technical\ report,\ research\ report,\ or\ white\ paper\ published\ by\ a\ university\ or\ some\ other\ institution.=Технический отчет, исследовательский отчет, или белая книга, выпущенная институтом или другим учреждением. An\ entry\ set\ is\ a\ group\ of\ entries\ which\ are\ cited\ as\ a\ single\ reference\ and\ listed\ as\ a\ single\ item\ in\ the\ bibliography.=Набор записей представляет собой группу записей, которые приводятся в виде единой ссылки и перечислены в виде одного элемента в библиографии. -Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. +Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. Supplemental\ material\ in\ a\ "Collection".=Дополнительные материалы в "Коллекции". Supplemental\ material\ in\ a\ "Periodical".\ This\ type\ may\ be\ useful\ when\ referring\ to\ items\ such\ as\ regular\ columns,\ obituaries,\ letters\ to\ the\ editor,\ etc.\ which\ only\ have\ a\ generic\ title.=Дополнительные материалы в "Периодическом издании". Этот тип может быть полезен при обращении к таким элементам, как обычные колонки, некрологи, письма к редактору и т.д., которые имеют только общее название. A\ thesis\ written\ for\ an\ educational\ institution\ to\ satisfy\ the\ requirements\ for\ a\ degree.=Тезис, написанный для учебного заведения с целью удовлетворения требований к степени. @@ -2280,10 +2345,6 @@ Removes\ digits.=Удаляет цифры. Presets=Предустановки -Check\ Proxy\ Setting=Проверить настройки прокси -Check\ connection=Проверить подключение -Connection\ failed\!=Соединение потеряно\! -Connection\ successful\!=Соединение успешно\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Создать группы из ключевых слов в следующем поле Generate\ groups\ for\ author\ last\ names=Создание групп для фамилий авторов Regular\ expression=Регулярное выражение @@ -2305,24 +2366,40 @@ Separate\ citations=Разделить цитаты Unprotect\ terms=Незащищенные термины Generate\ a\ new\ key\ for\ imported\ entries\ (overwriting\ their\ default)=Создать новый ключ для импортированных записей (взамен существующих) +Warn\ about\ duplicates\ on\ import=Предупреждать о дубликатах при импорте Import\ and\ Export=Импорт и экспорт Custom\ DOI\ URI=Пользовательский DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=Использовать пользовательский базовый URI от DOI для доступа к материалам Cited\ on\ pages=Цитаты на страницах +Please\ move\ the\ cursor\ into\ the\ document\ text.=Пожалуйста, переместите курсов в текст документа. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=Для получения информации о зрительном положении ваших цитат, мне необходимо передвинуть курсор, однако мне не удалось его захватить. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=Я не могу осуществить вставку в текущее положение курсора. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Пожалуйста, переместите курсор в положение нового цитирования. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Пожалуйста, создайте его в документе или измените в файле\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Пожалуйста, используйте последний вариант в файле оформления ниже во избежание проблем локализации. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' символа %0 является отображаемым именем для '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' символа %0 отсутствует в документе +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' абзаца %0 является отображаемым именем для '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' абзаца %0 отсутствует в документе +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Ошибка при проверке того, записывает ли Writer изменения или содержит записанные изменения. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Невозможно выполнить при включённом [Edit]/[Track Changes]/[Record]. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Изменения, выполненные JabRef, могут привести к неожиданным взаимодействиям с записанными изменениями. +Recording\ and/or\ Recorded\ changes=Изменения записываются и/или записаны +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Сперва используйте [Edit]/[Track Changes]/[Manage], чтобы их разрешить. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=Не удается найти действительный путь сертификации к запрошенной цели (%0), все равно загрузить? Download\ operation\ canceled.=Операция загрузки отменена. @@ -2340,7 +2417,6 @@ Entry\ Type=Тип записи Entry\ types=Типы записей Field\ names=Названия полей Others=Другие -Overwrite\ existing\ field\ values=Перезаписать текущие значения полей Recommended=Рекомендованный Authors\ and\ Title=Авторы и Заголовок @@ -2371,6 +2447,8 @@ Query=Запрос Question=Вопрос Select\ directory=Выберите каталог +Fulltext\ Index=Полнотекстовое индексирование +Automatically\ index\ all\ linked\ files\ for\ fulltext\ search=Автоматически индексировать все связанные файлы для полнотекстового поиска Rebuild\ fulltext\ search\ index=Перестроить полнотекствой индекс Rebuild\ fulltext\ search\ index\ for\ current\ library?=Перестроить полнотекстовой индекс для текущей библиотеки? Rebuilding\ fulltext\ search\ index...=Перестроить полнотекствой индекс... @@ -2383,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Удаленные сервисы Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Разрешить отправку PDF-файлов и цитат в онлайн сервис JabRef (Grobid) для определения метаданных. Это приводит к улучшению результатов. +Fetcher\ cannot\ be\ tested\!=Невозможно проверить сборщик данных\! +Fetcher\ unknown\!=Неизвестный сборщик данных\! Character\ by\ character=Посимвольно Embedded=Встроенный @@ -2397,7 +2477,31 @@ Could\ not\ extract\ Metadata\ from\:\ %0=Не удалось извлечь м Search\ across\ libraries\ in\ a\ new\ window=Поиск по библиотекам в новом окне Keep\ search\ string\ across\ libraries=Сохранить строку поиска по библиотекам Keep\ dialog\ always\ on\ top=Показывать диалог поверх всех окон - - - - +Search\ results\ from\ open\ libraries=Результаты поиска в открытых библиотеках + +Add\ certificate=Добавить сертификат +Serial\ number=Серийный номер +Issuer=Издатель +Valid\ from=Действителен с +Valid\ to=Действителен до +Signature\ algorithm=Алгоритм подписи +Version=Версия + +Error\ downloading=Ошибка загрузки + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Ошибка при записи метаданных. Смотрите журнал ошибок для подробностей. +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Не удалось записать метаданные, файл %1 не найден. +Success\!\ Finished\ writing\ metadata.=Успех\! Запись метаданных завершена. + +Custom\ API\ key=Пользовательский ключ API +Check\ %0\ API\ Key\ Setting=Проверьте настройку ключа API %0 + +Set=Задать +Append=Присоединить +Rename=Переименовать +New\ field\ name=Новое имя поля +Automatic\ field\ editor=Автоматический редактор поля + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Примечание\: Если в исходных записях отсутствуют ключевые слова, подходящие для конфигурации новой группы, то при подтверждении здесь они будут добавлены) +Assign=Назначить +Do\ not\ assign=Не назначать diff --git a/src/main/resources/l10n/JabRef_sv.properties b/src/main/resources/l10n/JabRef_sv.properties index 84779302d46..ae4d6c18d10 100644 --- a/src/main/resources/l10n/JabRef_sv.properties +++ b/src/main/resources/l10n/JabRef_sv.properties @@ -19,6 +19,7 @@ Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=Biblioteke %0/%1\ entries=%0/%1 poster + %0\ matches\ the\ regular\ expression\ %1=%0 matchar det reguljära uttrycket %1 %0\ matches\ the\ term\ %1=%0 matchar termen %1 @@ -138,8 +139,6 @@ Cite\ command=Citeringskommando Clear=Rensa -Clear\ fields=Rensa fält - Open\ /\ close\ entry\ editor=Öppna / stäng postredigerare Close\ dialog=Stäng dialog @@ -618,11 +617,18 @@ Remove\ selected\ entries\ from\ this\ group=Ta bort valda poster från denna gr Remove\ group=Ta bort grupp -Remove\ group\ "%0"?=Ta bort gruppen "%0"? +Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Ta bort gruppen "%0" och dess undergrupper? -Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. + + Remove\ link=Ta bort länk @@ -632,8 +638,6 @@ Remove\ string\ %0=Ta bort sträng %0 Removed\ group\ "%0".=Tog bort gruppen "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. - Removed\ string=Tog bort sträng Renamed\ string=Bytte namn på sträng @@ -685,9 +689,6 @@ Select\ entry\ type=Välj posttyp Select\ file\ from\ ZIP-archive=Välj fil från ZIP-arkiv -Set\ field=Sätt fält -Set\ fields=Sätt fält - Settings=Alternativ Shortcut=Genväg @@ -718,7 +719,6 @@ Size=Storlek Skipped\ -\ No\ PDF\ linked=Hoppade över - Ingen PDF länkad Skipped\ -\ PDF\ does\ not\ exist=Hoppade över - PDF fanns ej -Skipped\ entry.=Hoppade över post. source\ edit=ändring av källkod @@ -828,8 +828,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Rad %0 Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Rad %0\: Hittade korrupt referensnyckel %1 (kommatecken saknas). Download\ from\ URL=Ladda ned från URL Rename\ field=Byt namn på fält -Rename\ field\ to=Byt namn på fält till -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytta innehållet i ett fält till ett fält med ett annat namn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan inte använda port "%0" för fjärråtkomst; ett annat program kanske använder den. Försök med en annan port. @@ -1029,6 +1027,7 @@ Proxy\ requires\ authentication=Proxyn kräver autentisering Attention\:\ Password\ is\ stored\ in\ plain\ text\!=OBS\! Lösenordet sparas i klartext\! Clear\ connection\ settings=Rensa anslutningsinställningar + Open\ folder=Öppna mapp Export\ sort\ order=Soteringsordning vid export @@ -1370,7 +1369,6 @@ Protected\ terms\ file=Fil med skyddade termer Style\ file=Stilfil för OpenOffice/LibreOffice Open\ OpenOffice/LibreOffice\ connection=Öppna OpenOffice/LibreOffice-anslutning -You\ must\ enter\ at\ least\ one\ field\ name=Du måste ange minst ett fältnamn Non-ASCII\ encoded\ character\ found=Bokstäver som inte är ASCII-kodade hittades Toggle\ web\ search\ interface=Växla webbsökning @@ -1571,6 +1569,8 @@ Previous\ preview\ style=Föregående förhandsgranskningsstil + + diff --git a/src/main/resources/l10n/JabRef_tl.properties b/src/main/resources/l10n/JabRef_tl.properties index 0cb29b98e99..de37b3f268a 100644 --- a/src/main/resources/l10n/JabRef_tl.properties +++ b/src/main/resources/l10n/JabRef_tl.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 nag tugma sa regular na ekspresyon %1 %0\ matches\ the\ term\ %1=%0 tumugma sa termino %1 @@ -126,8 +127,6 @@ Cite\ command=Utos na nabanggit Clear=Malinaw -Clear\ fields=Malinaw na patlang - Close\ dialog=Isara ang dialog @@ -600,11 +599,18 @@ Remove\ selected\ entries\ from\ this\ group=Tanggalin ang napiling entries mula Remove\ group=Tanggaling ang grupo -Remove\ group\ "%0"?=Tanggalin ang grupo "%0"? +Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Tanggalin ang grupo "%0" at ang mababang grupo? -Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo +Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. + + Remove\ link=Tanggalin ang link @@ -613,8 +619,6 @@ Remove\ old\ entry=Tanggalin ang lumang entry Removed\ group\ "%0".=Tanggalin ang grupo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. - Removed\ string=Na tanggal ang string Renamed\ string=Palitan ulit ng pangalan ang string @@ -671,9 +675,6 @@ Select\ file\ from\ ZIP-archive=Piliin ang file mula sa na arkiba na ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Piliin ang nodes na puno para makita at matanggap o matanggihan ang pagbabago -Set\ field=Ihanda ang patlang -Set\ fields=Ihanda ang patlang - Settings=Mga settings Shortcut=Shortcut @@ -706,7 +707,6 @@ Size=Laki Skipped\ -\ No\ PDF\ linked=Nilaktawan - Walang PDF ang na linked Skipped\ -\ PDF\ does\ not\ exist=Nilaktawan - PDF ay hindi umiiral -Skipped\ entry.=Nilaktawan ang entry. source\ edit=i-edit ang pinagmulan @@ -820,10 +820,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=I-download mula sa URL Rename\ field=Palitan ang pangalan ng patlang -Append\ field=Ilagay ang field -Append\ to\ fields=Ilagay sa mga patlang -Rename\ field\ to=Palitan ang pangalan ng patlang sa -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Ilipat ang mga nilalaman ng isang patlang sa isang patlang na may ibang pangalan Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Hindi magagamit ang port %0 para sa remote na operasyon; maaaring gamitin ng isa pang application. Subukan ang pagtukoy ng isa pang port. @@ -1033,6 +1029,7 @@ Proxy\ requires\ authentication=Kinakailangan ng proxy ang pagpapatunay Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Pansin\: Ang password ay naka-imbak sa plain text\! Clear\ connection\ settings=I-clear ang mga setting ng koneksyon + Export\ sort\ order=Mag-export ng uri ng order Show\ extra\ columns=Magpakita ng mga dagdag na haligi @@ -1206,7 +1203,6 @@ Protected\ terms\ file=Protektadong mga tuntunin na file Style\ file=Estilo ng file Open\ OpenOffice/LibreOffice\ connection=Buksan ang OpenOffice/LibreOffice na koneksyon -You\ must\ enter\ at\ least\ one\ field\ name=Dapat kang magpasok ng hindi bababa sa isang pangalan ng field Non-ASCII\ encoded\ character\ found=Natagpuan ang naka-encode na karakter na hindi-ASCII Toggle\ web\ search\ interface=I-toggle ang interface ng paghahanap sa web @@ -1276,7 +1272,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Tinanggal ang l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Hindi makuha ang data mula sa entry '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Ang entry mula sa %0 ay hindi ma-parse. Invalid\ identifier\:\ '%0'.=Di-wastong identifier\: '%0'. -This\ paper\ has\ been\ withdrawn.=Ang papel na ito ay na-withdraw. Aux\ file=Aux file Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo na naglalaman ng mga entry na binanggit sa isang ibinigay na file ng TeX @@ -1364,6 +1359,8 @@ Default\ pattern=Default na pattern + + diff --git a/src/main/resources/l10n/JabRef_tr.properties b/src/main/resources/l10n/JabRef_tr.properties index 80d892c18af..fc09ebe0ce1 100644 --- a/src/main/resources/l10n/JabRef_tr.properties +++ b/src/main/resources/l10n/JabRef_tr.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 girdi -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 dışa aktarım başarılı. Kaydedilen dosyayı içeren klasör açılsın mı? + %0\ matches\ the\ regular\ expression\ %1=%0 şu Düzenli İfadeyle eşleşiyor %1 @@ -151,8 +151,6 @@ Cite\ command=Alıntı komutu Clear=Sil -Clear\ fields=Alanları sil - Open\ /\ close\ entry\ editor=Girdi düzenleyicisini aç/kapat Close\ dialog=Dialoğu kapat @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Seçili girdileri bu gruptan çıka Remove\ group=Grubu sil -Remove\ group\ "%0"?="%0" grubu silinsin mi? +Remove\ group\ and\ subgroups=Grubu ve altgrupları sil + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?="%0" grubu ve altgrupları silinsin mi? -Remove\ group\ and\ subgroups=Grubu ve altgrupları sil +Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. + + Remove\ link=Linki sil @@ -717,8 +722,6 @@ Remove\ string\ %0=%0 Dizgesini sil Removed\ group\ "%0".="%0" grubu silindi. -Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. - Removed\ string=Dizge silindi Renamed\ string=Dizge yeniden adlandırıldı @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=ZIP arşivinden dosyayı seçiniz Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Değişiklikleri görmek ve kabul ya da reddetmek için ağaç düğümlerini seçiniz -Set\ field=Alanı ata -Set\ fields=Alanları ata - Settings=Ayarlar Shortcut=Kısayol @@ -825,7 +825,6 @@ Size=Boyut Skipped\ -\ No\ PDF\ linked=Atlandı - PDF eklenmedi Skipped\ -\ PDF\ does\ not\ exist=Atlandı - PDF mevcut değil -Skipped\ entry.=Girdi atlandı. Sort\ subgroups=Alt grupları sırala @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Satır %0\: B No\ full\ text\ document\ found=Tam metin belge bulunamadı Download\ from\ URL=URL'den indir Rename\ field=Alanın adını değiştir -Append\ field=Alan ekle -Append\ to\ fields=Alanlara ekle -Rename\ field\ to=Alan adını şuna değiştir -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Alan içeriğini başka isimli bir alanın içine taşı Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Uzak operasyon için bağlantı noktası %0 kullanılamıyor; bir başka program kullanıyor olabilir. Başka bir bağlantı noktası deneyin. @@ -1243,6 +1238,15 @@ Use\ custom\ proxy\ configuration=Özelleştirilmiş vekil konfigürasyonu kulla Proxy\ requires\ authentication=Vekil sunucu kimlik denetleme gerektiriyor Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Dikkat\: Parola salt metin olarak kaydedildi\! Clear\ connection\ settings=Bağlantı ayarlarını sil +Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et +Check\ connection=Bağlantıyı kontrol et +Connection\ failed\!=Bağlantı başarısız\! +Connection\ successful\!=Bağlantı başarılı\! + +SSL\ Configuration=SSL Yapılandırması +SSL\ certificate\ file=SSL sertifika dosyası +Duplicate\ Certificates=Çift nüsha Sertifikalar +You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Open\ folder=Klasörü aç Export\ sort\ order=Sıralama kriterlerini dışa aktar @@ -1703,7 +1707,6 @@ Protected\ terms\ file=Korunmuş terimler dosyası Style\ file=Stil dosyası Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice bağlantısı aç -You\ must\ enter\ at\ least\ one\ field\ name=En az bir alan adı girmelisiniz Non-ASCII\ encoded\ character\ found=ASCII koduyla kodlanmamış karakter bulundu Non-UTF-8\ encoded\ field\ found=UTF-8 ile kodlanmamış alan bulundu Toggle\ web\ search\ interface=Ağ arama arayüzünü değiştir @@ -1841,7 +1844,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Alan içeriğin Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' dan girdi verileri alınamadı. Entry\ from\ %0\ could\ not\ be\ parsed.=%0'dan girdi çözümlenemedi. Invalid\ identifier\:\ '%0'.=Geçersiz tanımlayıcı\: '%0 '. -This\ paper\ has\ been\ withdrawn.=Bu yayın geri çekildi. empty\ citation\ key=boş atıf anahtarı Aux\ file=Aux dosya Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Belirli bir TeX dosyasında alıntılanmış girdileri içeren grup @@ -2324,10 +2326,6 @@ Removes\ digits.=Rakamları siler. Presets=Ön ayarlar -Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et -Check\ connection=Bağlantıyı kontrol et -Connection\ failed\!=Bağlantı başarısız\! -Connection\ successful\!=Bağlantı başarılı\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Aşağıdaki alandaki anahtar sözcüklerden gruplar oluştur Generate\ groups\ for\ author\ last\ names=Yazar soyadları için gruplar oluştur Regular\ expression=Düzenli ifade @@ -2400,7 +2398,6 @@ Entry\ Type=Girdi Türü Entry\ types=Girdi türleri Field\ names=Alan adları Others=Diğerleri -Overwrite\ existing\ field\ values=Mevcut alan değerlerinin üzerine yaz Recommended=Önerilen Authors\ and\ Title=Yazarlar ve Başlık @@ -2445,6 +2442,8 @@ Grobid\ URL=Grobid URL Remote\ services=Uzak servisler Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Metaverisini belirlemek için PDF dosyalarını ve ham atıf dizgelerini bir JabRef çevrim içi servisine (Grobid) göndermeye izin ver. Bu daha iyi sonuçlara yol açar. +Fetcher\ cannot\ be\ tested\!=Getirici test edilemiyor\! +Fetcher\ unknown\!=Getirici bilinmiyor\! Character\ by\ character=Karakter karakter Embedded=Gömülü @@ -2468,10 +2467,6 @@ Valid\ from=Geçerlilik tarihi Valid\ to=Son kullanma tarihi Signature\ algorithm=İmza algoritması Version=Sürüm -SSL\ Configuration=SSL Yapılandırması -SSL\ certificate\ file=SSL sertifika dosyası -Duplicate\ Certificates=Çift nüsha Sertifikalar -You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Error\ downloading=İndirmede hata @@ -2479,3 +2474,7 @@ Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Metaverisi Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metaverisi yazma başarısız, %1 dosyası bulunamadı. Success\!\ Finished\ writing\ metadata.=Başarılı\! Metaverisi yazma bitirildi. +Custom\ API\ key=Özel API anahtarı +Check\ %0\ API\ Key\ Setting=%0 API Anahtar Ayarlarını Kontrol Edin + + diff --git a/src/main/resources/l10n/JabRef_vi.properties b/src/main/resources/l10n/JabRef_vi.properties index 26aaa90bf04..d82df8e5422 100644 --- a/src/main/resources/l10n/JabRef_vi.properties +++ b/src/main/resources/l10n/JabRef_vi.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 khớp biểu thức chính tắc %1 %0\ matches\ the\ term\ %1=%0 khớp thuật ngữ %1 @@ -133,8 +134,6 @@ Cite\ command=Lệnh trích dẫn Clear=Xóa -Clear\ fields=Xóa các dữ liệu - Close\ dialog=Đóng hộp thoại @@ -612,11 +611,18 @@ Remove\ entry\ from\ import=Loại bỏ mục khỏi lệnh nhập Remove\ group=Loại bỏ nhóm -Remove\ group\ "%0"?=Loại bỏ nhóm "%0"? +Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Loại bỏ nhóm "%0" và các nhóm con của nó? -Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con +Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. + + Remove\ link=Loại bỏ liên kết @@ -625,8 +631,6 @@ Remove\ old\ entry=Loại bỏ mục cũ Removed\ group\ "%0".=Đã loại bỏ nhóm "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. - Removed\ string=Đã loại bỏ chuỗi Renamed\ string=Chuỗi được đặt tên lại @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Chọn tập tin từ tập tin ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Chọn các nốt trên sơ đồ hình cây để xem và chấp nhận hoặc từ chối thay đổi -Set\ field=Thiết lập dữ liệu -Set\ fields=Thiết lập các dữ liệu - Settings=Các thiết lập Shortcut=Phím tắt @@ -712,7 +713,6 @@ Size=Kích thước Skipped\ -\ No\ PDF\ linked=Bị bỏ qua - Không có tập tin PDF được liên kết Skipped\ -\ PDF\ does\ not\ exist=Bỏ qua - tập tin PDF không tồn tại -Skipped\ entry.=Mục bị bỏ qua. source\ edit=chỉnh sửa nguồn @@ -826,8 +826,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Kéo từ URL Rename\ field=Đổi tên dữ liệu -Rename\ field\ to=Đổi tên dữ liệu thành -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Di chuyển nội dung của một dữ liệu sang một dữ liệu có tên khác Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Không thể dùng cổng %0 cho lệnh chạy từ xa; một ứng dụng khác có thể đang dùng nó. Hãy thử chỉ định một cổng khác. @@ -900,6 +898,7 @@ Finished\ automatically\ setting\ external\ links.=Thiết lập tự động c + Open\ folder=Mở thư mục @@ -1135,6 +1134,8 @@ Default\ pattern=Kiểu mặc định + + diff --git a/src/main/resources/l10n/JabRef_zh_CN.properties b/src/main/resources/l10n/JabRef_zh_CN.properties index 0577de6a7b9..baca5d6da46 100644 --- a/src/main/resources/l10n/JabRef_zh_CN.properties +++ b/src/main/resources/l10n/JabRef_zh_CN.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 条目 +Export\ operation\ finished\ successfully.=导出成功。 + +Reveal\ in\ File\ Explorer=在文件夹中显示 %0\ matches\ the\ regular\ expression\ %1=%0 匹配正则表达式 %1 @@ -28,7 +31,7 @@ Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (DEFAULT\ abbreviation)= Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (MEDLINE\ abbreviation)=缩写选中记录的期刊名 (MEDLINE 格式缩写) Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (SHORTEST\ UNIQUE\ abbreviation)=缩写选中记录的期刊名 (最短唯一 (SHORTEST UNIQUE) 格式缩写) -Abbreviate\ names=缩写名 +Abbreviate\ names=缩写名(J. Smith) Abbreviated\ %0\ journal\ names.=缩写的 %0 期刊名称。 Abbreviation=缩写 @@ -150,8 +153,6 @@ Cite\ command=引用命令 Clear=清除 -Clear\ fields=清除字段内容 - Open\ /\ close\ entry\ editor=打开/关闭条目编辑器 Close\ dialog=关闭对话框 @@ -201,7 +202,7 @@ Could\ not\ run\ the\ 'vim'\ program.=无法运行 'vim' 程序。 Could\ not\ save\ file.=无法保存文件 Character\ encoding\ '%0'\ is\ not\ supported.=,不支持编码 '%0'。 -Create\ custom\ fields\ for\ each\ BibTeX\ entry=为每个 BibTeX 条目创建自定义字段 +Create\ custom\ fields\ for\ each\ BibTeX\ entry=Create custom fields for each BibTeX entry crossreferenced\ entries\ included=包含交叉引用的记录 @@ -265,7 +266,7 @@ Display\ help\ on\ command\ line\ options=用命令行选项显示帮助 Display\ only\ entries\ belonging\ to\ all\ selected\ groups=仅显示所选分组的交集条目 Display\ version=显示版本 -Do\ not\ abbreviate\ names=不要缩写姓名 +Do\ not\ abbreviate\ names=不缩写名(John Smith) Do\ not\ import\ entry=不导入记录 @@ -376,9 +377,9 @@ First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=首先选 Fit\ table\ horizontally\ on\ screen=列表宽度填满屏幕宽度 Float=浮动 (结果上浮到最前) -Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式\: 选项卡\: 字段; 字段; ...(例如 General\:url;pdf;note...) +Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式:选项卡\:字段;字段;...(例如:General\:url;pdf;note...) -Format\ of\ author\ and\ editor\ names=作者和编者的姓名格式 +Format\ of\ author\ and\ editor\ names=author和editor的姓名格式 Format\ string=格式化简写字串 Format\ used=使用的格式 @@ -492,7 +493,7 @@ Key\ bindings=热键 Key\ bindings\ changed=热键已更改 -Key\ pattern=键值表达式 +Key\ pattern=Key pattern keys\ in\ library=文献库中的键值 @@ -541,7 +542,9 @@ Moved\ group\ "%0".=移动了分组 "%0"。 Mr.\ DLib\ Privacy\ settings=Mr. DLib 隐私设置 +No\ database\ is\ open=No database is open +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=We need a database to export from. Open one. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=没有收到 Mr. DLib 对于本条目的建议。 @@ -549,9 +552,9 @@ Error\ while\ fetching\ recommendations\ from\ Mr.DLib.=在接收 Mr. DLib 的 Name=名称 -Name\ formatter=姓名格式化器 +Name\ formatter=姓名格式化 -Natbib\ style=Natbib 格式 +Natbib\ style=Natbib样式 nested\ AUX\ files=nested AUX 文件 @@ -700,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=从这个组里移除选定记录 Remove\ group=移除分组 -Remove\ group\ "%0"?=移除分组 "%0"? +Remove\ group\ and\ subgroups=移除分组和子分组 + +Remove\ groups\ and\ subgroups=移除分组和子分组 + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=移除所有选定的分组但保留其子分组? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=移除分组 "%0" 但保留其子分组? + +Remove\ groups=移除分组 + +Removed\ all\ selected\ groups.=移除选中的分组 Remove\ group\ "%0"\ and\ its\ subgroups?=移除分组 "%0" 和它的子分组? -Remove\ group\ and\ subgroups=移除分组和子分组 +Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=移除所有选定的分组及其子分组? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=移除所有选定的分组及其子分组 Remove\ link=移除链接 @@ -714,8 +731,6 @@ Remove\ string\ %0=移除字符串 %0 Removed\ group\ "%0".=已移除分组 "%0"。 -Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 - Removed\ string=已移除简写字串 Renamed\ string=重命名简写字串 @@ -753,7 +768,7 @@ Review=评论 Review\ changes=复查修改 Review\ Field\ Migration=查看字段迁移 -Loading=加载中 +Loading=启动 Save=保存 Save\ all\ finished.=完成保存全部。 @@ -764,10 +779,10 @@ Save\ before\ closing=关闭前保存 Save\ library=保存 Save\ library\ as...=另存为 -Saving=保存中 +Saving=保存 Saving\ all\ libraries...=正在保存所有数据库... Saving\ library=正在保存文献库 -Library\ saved=Library saved +Library\ saved=文献库已保存 Saved\ selected\ to\ '%0'.=保存选中到 '%0'. Search=查找 @@ -787,24 +802,21 @@ Select\ file\ from\ ZIP-archive=从 ZIP-压缩包中选择文件 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=选择树节点查看和接受/拒绝修改 -Set\ field=设置字段内容 -Set\ fields=设置字段内容 - Settings=设置 Shortcut=快捷键 Show/edit\ %0\ source=显示/编辑 %0 源代码 -Show\ 'Firstname\ Lastname'=显示 '名 (Firstname) 姓 (Lastname)' +Show\ 'Firstname\ Lastname'=名在前,姓在后(John Smith) -Show\ 'Lastname,\ Firstname'=显示 '姓 (Lastname), 名 (Firstname)' +Show\ 'Lastname,\ Firstname'=姓在前,名在后(Smith, John) Show\ BibTeX\ source\ by\ default=默认显示 BibTeX 源代码 Show\ confirmation\ dialog\ when\ deleting\ entries=删除多条记录时发出警告 -Show\ last\ names\ only=只显示“姓 (Lastname)” +Show\ last\ names\ only=仅显示姓(Smith) Show\ names\ unchanged=显示原始姓名字串 @@ -822,12 +834,17 @@ Size=大小 Skipped\ -\ No\ PDF\ linked=跳过 - 没有 PDF 链接 Skipped\ -\ PDF\ does\ not\ exist=跳过 - PDF 不存在 -Skipped\ entry.=已跳过记录 +JabRef\ skipped\ the\ entry.=JabRef 跳过了条目。 +Import\ error=导入错误 +Open\ library\ error=打开文献库出错 +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=请检查您的库文件是否存在语法错误。 +SourceTab\ error=SourceTab错误 +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=通过条目编辑器的 "{}bibtex source" 标签中输入时出错。 Sort\ subgroups=排序子组 source\ edit=源代码编辑 -Special\ name\ formatters=特殊的姓名格式化器 +Special\ name\ formatters=姓名格式化 Statically\ group\ entries\ by\ manual\ assignment=手动创建静态分组 @@ -956,7 +973,7 @@ Refuse\ to\ save\ the\ library\ before\ external\ changes\ have\ been\ reviewed. Library\ protection=文献库保护 Unable\ to\ save\ library=无法保存文献库 -Citation\ key\ generator=Citationkey生成器 +Citation\ key\ generator=Citationkey Unable\ to\ open\ link.=无法打开链接。 MIME\ type=MIME 类型 @@ -969,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=第 %0 行\: No\ full\ text\ document\ found=未发现完整的文档 Download\ from\ URL=从 URL 下载 Rename\ field=重命名字段 -Append\ field=添加字段 -Append\ to\ fields=追加到字段 -Rename\ field\ to=重命名字段为 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=将某个字段的内容移动到另一个字段中 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=无法使用端口 %0 进行远程操作;该端口可能被其它应用程序占用,请使用其它端口。 @@ -1042,6 +1055,7 @@ Cite\ special=引用特殊 Extra\ information\ (e.g.\ page\ number)=额外信息(例如:页码) Manage\ citations=管理文献引用 Problem\ modifying\ citation=修改文献引用存在问题 +Problem\ collecting\ citations=Problem collecting citations Citation=文献引用 Connecting...=连接中... Select\ style=选择引用样式 @@ -1056,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=同步 OpenOffice/LibreOffice 参考 Select\ which\ open\ Writer\ document\ to\ work\ on=选择使用哪个打开的 Writer 文档 Connected\ to\ document=连接到文档 +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Could not connect to any Writer document. Please make sure a document is open before using the 'Select Writer document' button to connect to it. +No\ Writer\ documents\ found=No Writer documents found Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=插入一条没有内容的引用(这条记录将会出现在引用列表中) Cite\ selected\ entries\ with\ extra\ information=引用包含额外信息的选中记录 @@ -1064,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=保证参考文献是最新的 Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=您的OpenOffice/LibreOffice文档引用了 citationkey "%0",但未在当前库中找到。 +This\ operation\ requires\ a\ bibliography\ database.=This operation requires a bibliography database. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Your OpenOffice/LibreOffice document references at least %0 citation keys which could not be found in your current library. Some of these are %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=Your OpenOffice/LibreOffice document references no citation keys which could also be found in your current library. Unable\ to\ synchronize\ bibliography=无法同步参考文献 Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=合并仅仅由空格分隔的两段引文 @@ -1073,7 +1092,12 @@ Autodetection\ failed=自动检测失败 Please\ wait...=请稍候... Connection\ lost=连接丢失 +Could\ not\ update\ bibliography=Could not update bibliography +Not\ connected\ to\ document=Not connected to document +Problem\ combining\ cite\ markers=Problem combining cite markers +Unable\ to\ reload\ style\ file=Unable to reload style file +Problem\ during\ separating\ cite\ markers=Problem during separating cite markers Automatically\ sync\ bibliography\ when\ inserting\ citations=当插入文献引用时自动同步参考文献 Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=在当前标签查找 BibTeX 记录 @@ -1176,6 +1200,7 @@ Toggle\ relevance=标记为相关 Toggle\ quality\ assured=标记为质量已确认 Toggle\ print\ status=标记打印状态 Update\ keywords=更新关键词 +Problem\ connecting=Problem connecting Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=与OpenOffice/LibreOffice的连接已丢失。请确保OpenOffice/LibreOffice正在运行,并尝试重新连接。 JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef 将会为每个条目发送至少一个请求至发布者。 @@ -1228,6 +1253,16 @@ Use\ custom\ proxy\ configuration=使用自定义 HTTP 代理 Proxy\ requires\ authentication=代理服务器需要密码 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意\: 密码以明文形式保存\! Clear\ connection\ settings=清除连接设置 +Check\ Proxy\ Setting=检查代理设置 +Check\ connection=检查连接 +Connection\ failed\!=连接失败! +Connection\ successful\!=连接成功! + +SSL\ Configuration=SSL配置 +SSL\ configuration\ changed=SSL配置已更改 +SSL\ certificate\ file=SSL证书文件 +Duplicate\ Certificates=复制证书 +You\ already\ added\ this\ certificate=您已经添加了此证书 Open\ folder=打开文件夹 Export\ sort\ order=导出顺序 @@ -1264,6 +1299,7 @@ Please\ open\ %0\ manually.=请手动打开 %0。 The\ link\ has\ been\ copied\ to\ the\ clipboard.=链接已复制到剪贴板 Open\ %0\ file=打开文件 %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=无法自动检测终端。请在首选项中自定义一个终端。 Cannot\ delete\ file=无法删除文件 File\ permission\ error=文件权限错误 @@ -1365,7 +1401,7 @@ Automatically\ setting\ file\ links=自动设置文件链接 Regenerating\ citation\ keys\ according\ to\ metadata=根据元数据重新生成 citationkey Regenerate\ all\ keys\ for\ the\ entries\ in\ a\ BibTeX\ file=重新生成 BibTeX 文件中所有记录的键值 Show\ debug\ level\ messages=显示调试级别消息 -Default\ library\ mode=Default library mode +Default\ library\ mode=默认文献库模式 Show\ only\ preferences\ deviating\ from\ their\ default\ value=只显示与默认值不同的首选项 default=默认 key=键值 @@ -1570,10 +1606,21 @@ Custom=自定义 Export\ cited=导出已被引用的 Unable\ to\ generate\ new\ library=无法生成新库 +The\ cursor\ is\ in\ a\ protected\ area.=The cursor is in a protected area. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=No bibliography database is open for citation. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=No database is open for updating citation markers after citing. +No\ bibliography\ entries\ are\ selected\ for\ citation.=No bibliography entries are selected for citation. +No\ bibliography\ style\ is\ selected\ for\ citation.=No bibliography style is selected for citation. +No\ database=No database +No\ entries\ selected\ for\ citation=No entries selected for citation +Open\ one\ before\ citing.=Open one before citing. +Select\ one\ before\ citing.=Select one before citing. +Select\ some\ before\ citing.=Select some before citing. Found\ identical\ ranges=Found identical ranges Found\ overlapping\ ranges=Found overlapping ranges @@ -1621,7 +1668,7 @@ Remove\ list=删除列表 Add\ abbreviation=添加缩写 Full\ journal\ name=期刊全称 Abbreviation\ name=缩写名 -Shortest\ unique\ abbreviation=Shortest unique abbreviation +Shortest\ unique\ abbreviation=最简缩写 No\ abbreviation\ files\ loaded=缩写文件未加载 @@ -1640,6 +1687,8 @@ Issue\ report\ successful=问题报告成功 Your\ issue\ was\ reported\ in\ your\ browser.=您的问题已在浏览器中报告。 The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=日志和异常信息已复制到剪贴板中。 Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=请将此信息粘贴至问题描述中 (用 Ctrl + V)。 +Last\ notification=最新通知 +Check\ the\ event\ log\ to\ see\ all\ notifications=在事件日志中查看所有通知 Host=主机 Port=端口 @@ -1677,7 +1726,6 @@ Protected\ terms\ file=受保护的术语文件 Style\ file=样式文件 Open\ OpenOffice/LibreOffice\ connection=打开 OpenOffice/LibreOffice 连接 -You\ must\ enter\ at\ least\ one\ field\ name=您至少需要输入一个字段名 Non-ASCII\ encoded\ character\ found=发现Non-ASCII编码字符 Non-UTF-8\ encoded\ field\ found=未找到 UTF-8 编码的字段 Toggle\ web\ search\ interface=切换网页搜索面板 @@ -1758,7 +1806,7 @@ Unhandled\ exception\ occurred.=发生了无法处理的异常。 strings\ included=包含的字符串 Escape\ underscores=转义下划线 -Color=色彩 +Color=颜色 Please\ also\ add\ all\ steps\ to\ reproduce\ this\ issue,\ if\ possible.=如果可能的话,也请添加所有步骤以重现这个问题。 Fit\ width=适应宽度 Fit\ a\ single\ page=适应单页 @@ -1815,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=移除字段内 Could\ not\ retrieve\ entry\ data\ from\ '%0'.=无法从 "%0" 中检索条目数据。 Entry\ from\ %0\ could\ not\ be\ parsed.=无法解析 %0 中的条目。 Invalid\ identifier\:\ '%0'.=无效的标识符:'%0'。 -This\ paper\ has\ been\ withdrawn.=这篇论文已被撤回。 empty\ citation\ key=Citationkey为空 Aux\ file=Aux 文件 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=包含给定TeX文件中引用的条目组 @@ -1883,7 +1930,7 @@ View\ change\ log=查看变更记录 View\ event\ log=查看事件日志 Website=网站 -Override\ default\ font\ settings=跳过默认字体设置 +Override\ default\ font\ settings=覆盖默认的字体设置 Clear\ search=清除搜索 Click\ help\ to\ learn\ about\ the\ migration\ of\ pre-3.6\ databases.=点击帮助了解 pre-3.6 数据库的迁移情况。 @@ -1993,47 +2040,47 @@ Import\ entries\ from\ LaTeX\ files=Import entries from LaTeX files Import\ new\ entries=Import new entries Group\ color=分组颜色 -Columns=Columns +Columns=列 File\ type=File type IEEE=IEEE Internal=Internal Special=Special Remove\ column=Remove column -Add\ custom\ column=Add custom column -Update\ to\ current\ column\ order=Update to current column order -Sort\ column\ one\ step\ upwards=Sort column one step upwards -Sort\ column\ one\ step\ downwards=Sort column one step downwards +Add\ custom\ column=添加自定义列 +Update\ to\ current\ column\ order=更新到当前的列顺序 +Sort\ column\ one\ step\ upwards=上移一位 +Sort\ column\ one\ step\ downwards=下移一位 List\ must\ not\ be\ empty.=List must not be empty. -Order=Order +Order=排序 -Add\ field\ to\ filter\ list=Add field to filter list +Add\ field\ to\ filter\ list=添加字段到过滤列表 Add\ formatter\ to\ list=Add formatter to list Filter\ List=Filter List Open\ files...=打开文件... Affected\ fields\:=补全字段 -Show\ preview\ as\ a\ tab\ in\ entry\ editor=Show preview as a tab in entry editor -Font=Font -Visual\ theme=Visual theme +Show\ preview\ as\ a\ tab\ in\ entry\ editor=在条目编辑器中以标签形式显示预览 +Font=字体 +Visual\ theme=主题风格 Light\ theme=明亮主题 Dark\ theme=暗色主题 Custom\ theme=自定义 Overwrite\ existing\ keys=覆盖该字段的内容 -Key\ patterns=不同类型条目的键值风格 +Key\ patterns=Citationkey样式 Font\ settings=Font settings Please\ specify\ a\ css\ theme\ file.=请指定一个 css 主题文件。 You\ must\ enter\ an\ integer\ value\ higher\ than\ 8.=You must enter an integer value higher than 8. Letters\ after\ duplicate\ generated\ keys=为重复的Citationkey添加字母后缀 -Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 A (a, b, ...) -Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 B (b, c, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 a (a, b, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 b (b, c, ...) Always\ add\ letter\ (a,\ b,\ ...)\ to\ generated\ keys=在生成键值时总是添加字母 (a, b, ...) -Default\ pattern=默认模式 +Default\ pattern=Default pattern Reset\ %s\ to\ default\ value=Reset %s to default value Library\ mode=Library mode Reset\ to\ recommended=Reset to recommended Remove\ all=Remove all -Reset\ All=Reset All +Reset\ All=全部重置 Linked\ identifiers=Linked identifiers insert\ entries=insert entries @@ -2056,13 +2103,14 @@ Explicit\ selection=手动选择 Searching\ for\ a\ keyword=指定字段的关键词 Free\ search\ expression=检索语句 Specified\ keywords=字段内容 -Cited\ entries=Cited entries +Cited\ entries=根据AUX文件 Search\ term\ is\ empty.=Search term is empty. Invalid\ regular\ expression.=无效的正则表达式 Please\ provide\ a\ valid\ aux\ file.=请提供一个有效的AUX文件 Keyword\ delimiter=Keyword delimiter Hierarchical\ keyword\ delimiter=Hierarchical keyword delimiter Escape\ ampersands=转义逻辑"与"符号 +Escape\ dollar\ sign=Escape dollar sign Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Hint\:\n\nTo search all fields for Smith, enter\:\nsmith\n\nTo search the field author for Smith and the field title for electrical, enter\:\nauthor\=Smith and title\=electrical @@ -2272,7 +2320,7 @@ Reveal\ in\ file\ explorer=Reveal in file explorer Autolink\ files=自动链接文件 -Custom\ editor\ tabs=条目编辑器的标签 +Custom\ editor\ tabs=条目编辑器 Custom\ export\ formats=自定义导出格式 Custom\ import\ formats=自定义导入格式 @@ -2283,7 +2331,7 @@ Customized\ preview\ style=自定义预览样式 Next\ preview\ style=下一个预览样式 Previous\ preview\ style=上一个预览样式 -(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=( Note\: Press return to commit changes in the table\! ) +(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=(注意:按回车键以确认更改) Reset=Reset New\ inproceedings=新的inproceedings Reset\ entry\ types\ and\ fields\ to\ defaults=Reset entry types and fields to defaults @@ -2297,10 +2345,6 @@ Removes\ digits.=Removes digits. Presets=默认 -Check\ Proxy\ Setting=检查代理设置 -Check\ connection=检查连接 -Connection\ failed\!=连接失败! -Connection\ successful\!=连接成功! Generate\ groups\ from\ keywords\ in\ the\ following\ field=根据以下字段的关键字生成组 Generate\ groups\ for\ author\ last\ names=根据作者的姓氏生成组 Regular\ expression=正则表达式 @@ -2328,19 +2372,34 @@ Custom\ DOI\ URI=自定义 DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=使用基于DOI的URI访问文章 Cited\ on\ pages=Cited on pages +Please\ move\ the\ cursor\ into\ the\ document\ text.=Please move the cursor into the document text. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=To get the visual positions of your citations I need to move the cursor around, but could not get it. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=I cannot insert to the cursors current location. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Please move the cursor to the location for the new citation. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Please create it in the document or change in the file\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Please use the latter in the style file below to avoid localization problems. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 character style '%1' is a display name for '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 character style '%1' is missing from the document +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 paragraph style '%1' is a display name for '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 paragraph style '%1' is missing from the document +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Error while checking if Writer is recording changes or has recorded changes. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Cannot work with [Edit]/[Track Changes]/[Record] turned on. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Changes by JabRef could result in unexpected interactions with recorded changes. +Recording\ and/or\ Recorded\ changes=Recording and/or Recorded changes +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Use [Edit]/[Track Changes]/[Manage] to resolve them first. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=无法找到请求目标(%0)的有效证书路径,仍然下载? Download\ operation\ canceled.=下载操作已取消。 @@ -2358,7 +2417,6 @@ Entry\ Type=条目类型 Entry\ types=条目类型 Field\ names=字段名 Others=其他 -Overwrite\ existing\ field\ values=覆盖现有字段内容 Recommended=推荐 Authors\ and\ Title=作者和标题 @@ -2401,8 +2459,10 @@ Found\ matches\ in\ Annotations\:=在注释中找到匹配项: Grobid\ URL=Grobid URL Remote\ services=远程服务 -Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Allow sending PDF files and raw citation strings to a JabRef online service (Grobid) to determine Metadata. This produces better results. +Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=允许将 PDF 文件和原始引用发送到JabRef在线服务(Grobid)以获取元数据 +Fetcher\ cannot\ be\ tested\!=Fetcher cannot be tested\! +Fetcher\ unknown\!=Fetcher unknown\! Character\ by\ character=Character by character Embedded=内嵌 @@ -2419,6 +2479,29 @@ Keep\ search\ string\ across\ libraries=在所有库中保留搜索语句 Keep\ dialog\ always\ on\ top=保持窗口位于最前面 Search\ results\ from\ open\ libraries=Search results from open libraries +Add\ certificate=添加证书 +Serial\ number=序号 +Issuer=签发者 +Valid\ from=起始有效期 +Valid\ to=截至有效期 +Signature\ algorithm=签名算法 +Version=版本 + +Error\ downloading=下载出错 + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=写入元数据时出错。详细信息请参阅错误日志。 +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Failed to write metadata, file %1 not found. +Success\!\ Finished\ writing\ metadata.=Success\! Finished writing metadata. +Custom\ API\ key=自定义API +Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting +Set=设定 +Append=附加 +Rename=重命名 +New\ field\ name=新的字段名 +Automatic\ field\ editor=自动化条目编辑 +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note\: If original entries lack keywords to qualify for the new group configuration, confirming here will add them) +Assign=Assign +Do\ not\ assign=Do not assign diff --git a/src/main/resources/l10n/JabRef_zh_TW.properties b/src/main/resources/l10n/JabRef_zh_TW.properties index 588f8dd4d77..288532048c0 100644 --- a/src/main/resources/l10n/JabRef_zh_TW.properties +++ b/src/main/resources/l10n/JabRef_zh_TW.properties @@ -15,6 +15,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 條目 + %0\ matches\ the\ regular\ expression\ %1=%0 符合正規表達式 %1 %0\ matches\ the\ term\ %1=%0 符合字詞 %1 @@ -142,8 +143,6 @@ Cite\ command=引用指令 Clear=清除 -Clear\ fields=清除欄位 - Close\ dialog=關閉對話視窗 @@ -579,6 +578,13 @@ Remove\ group=移除群組 + + + + + + + Remove\ link=移除連結 Remove\ old\ entry=移除舊有條目 @@ -588,7 +594,6 @@ Remove\ string\ %0=移除字串 %0 - Replace=取代 Replace\ With\:=取代為: Find\:=尋找: @@ -627,7 +632,6 @@ Select\ all=全選 - Settings=設定 Shortcut=快速鍵 @@ -728,8 +732,6 @@ Unable\ to\ save\ library=無法儲存文獻倉庫 Rename\ field=重新命名欄位 -Append\ field=添加欄位 -Append\ to\ fields=添加到欄位 Autosave\ local\ libraries=自動儲存本機文獻倉庫 @@ -806,6 +808,11 @@ Proxy\ configuration=代理伺服器設定 Use\ custom\ proxy\ configuration=使用自訂代理伺服器 Proxy\ requires\ authentication=代理伺服器需要認證 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意:密碼將以明文形式存放! +Check\ Proxy\ Setting=檢查代理設定 +Check\ connection=檢查連接 +Connection\ failed\!=連線失敗! +Connection\ successful\!=連線成功! + Export\ sort\ order=匯出順序 @@ -984,7 +991,7 @@ Set\ rank\ to\ five=設定評分為 5 級 Order=順序 -Affected\ fields\:=影響欄位\: +Affected\ fields\:=影響欄位\: Show\ preview\ as\ a\ tab\ in\ entry\ editor=在條目編輯器中以分頁形式顯示預覽 Font=字型 Visual\ theme=界面主題 @@ -1056,10 +1063,6 @@ Remove\ digits=移除數字 Removes\ digits.=移除數字 -Check\ Proxy\ Setting=檢查代理設定 -Check\ connection=檢查連接 -Connection\ failed\!=連線失敗! -Connection\ successful\!=連線成功! Generate\ groups\ for\ author\ last\ names=依據作者姓氏生成群組 Regular\ expression=正規表達式 @@ -1102,3 +1105,5 @@ Select\ directory=選擇資料夾 + + diff --git a/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..75a1a176597 --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java @@ -0,0 +1,105 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.copyormovecontent.CopyOrMoveFieldContentTabViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class CopyOrMoveFieldContentTabViewModelTest { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998"); + bibDatabase = new BibDatabase(); + copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + } + + @Test + void copyValueDoesNotCopyBlankValues() { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + copyOrMoveFieldContentTabViewModel.copyValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.DATE), "YEAR field is not copied correctly to the DATE field"); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR), "YEAR field should not have changed"); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE), "DATE field should not have changed because the YEAR field is blank e.g it doesn't exist"); + } + + @Test + void swapValuesShouldNotSwapFieldValuesIfOneOfTheValuesIsBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE)); + assertEquals(Optional.empty(), entryB.getField(StandardField.YEAR)); + } + + @Test + void swapValuesShouldSwapFieldValuesIfBothValuesAreNotBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(List.of(Optional.of("2014"), Optional.of("2015")), + List.of(entryA.getField(StandardField.YEAR), entryA.getField(StandardField.DATE)), + "YEAR and DATE values didn't swap"); + } + + @Test + void moveValueShouldNotMoveValueIfToFieldIsNotBlankAndOverwriteIsNotEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(false); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + } + + @Test + void moveValueShouldMoveValueIfOverwriteIsEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + private CopyOrMoveFieldContentTabViewModel newTwoFieldsViewModel(BibEntry... selectedEntries) { + return new CopyOrMoveFieldContentTabViewModel(List.of(selectedEntries), bibDatabase, stateManager); + } +} diff --git a/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..1221580762a --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java @@ -0,0 +1,118 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.editfieldcontent.EditFieldContentViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotEquals; +import static org.mockito.Mockito.mock; + +public class EditFieldContentTabViewModelTest { + EditFieldContentViewModel editFieldContentViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, ""); + + bibDatabase = new BibDatabase(); + editFieldContentViewModel = new EditFieldContentViewModel(bibDatabase, List.of(entryA, entryB), stateManager); + } + + @Test + void clearSelectedFieldShouldClearFieldContentEvenWhenOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.YEAR)); + } + + @Test + void clearSelectedFieldShouldDoNothingWhenFieldDoesntExistOrIsEmpty() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.FILE); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.FILE)); + } + + @Test + void setFieldValueShouldNotDoAnythingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfFieldContentIsEmpty() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryB.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldDoNothingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldAppendFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("20150"), entryA.getField(StandardField.YEAR)); + } + + @Test + void getAllFieldsShouldNeverBeEmpty() { + assertNotEquals(0, editFieldContentViewModel.getAllFields().size()); + } + + @Test + void getSelectedFieldShouldHaveADefaultValue() { + assertNotEquals(null, editFieldContentViewModel.getSelectedField()); + } +} diff --git a/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java new file mode 100644 index 00000000000..bf5a246e97f --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java @@ -0,0 +1,111 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.renamefield.RenameFieldViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.field.UnknownField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class RenameFieldViewModelTest { + RenameFieldViewModel renameFieldViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014") + .withField(StandardField.AUTHOR, "Doe"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, "") + .withField(StandardField.AUTHOR, "Eddie"); + + bibDatabase = new BibDatabase(); + renameFieldViewModel = new RenameFieldViewModel(List.of(entryA, entryB), bibDatabase, stateManager); + } + + @Test + void renameFieldShouldRenameFieldIfItExist() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName("ETAD"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryA.getField(StandardField.DATE)); + + assertEquals(Optional.of("1998"), entryB.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + @Test + void renameFieldShouldDoNothingIfFieldDoNotExist() { + Field toRenameField = new UnknownField("Some_field_that_doesnt_exist"); + renameFieldViewModel.selectField(toRenameField); + renameFieldViewModel.setNewFieldName("new_field_name"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.empty(), entryA.getField(toRenameField)); + assertEquals(Optional.empty(), entryA.getField(new UnknownField("new_field_name"))); + + assertEquals(Optional.empty(), entryB.getField(toRenameField)); + assertEquals(Optional.empty(), entryB.getField(new UnknownField("new_field_name"))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameIsEmpty() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName(""); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField(""))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField(""))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameHasWhitespaceCharacters() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName("Hello, World"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField("Hello, World"))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField("Hello, World"))); + } + + @Test + void renameFieldShouldDoNothingWhenThereIsAlreadyAFieldWithTheSameNameAsNewFieldName() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName(StandardField.YEAR.getName()); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + } +} diff --git a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java index 1f1c20fc329..9fded7c7802 100644 --- a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java +++ b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java @@ -56,7 +56,7 @@ void compareWithChangedGroup() { Optional groupDiff = GroupDiff.compare(originalMetaData, newMetaData); - Optional expectedGroupDiff = Optional.of(new GroupDiff(newMetaData.getGroups().get(), originalMetaData.getGroups().get())); + Optional expectedGroupDiff = Optional.of(new GroupDiff(originalMetaData.getGroups().get(), newMetaData.getGroups().get())); assertEquals(expectedGroupDiff.get().getNewGroupRoot(), groupDiff.get().getNewGroupRoot()); assertEquals(expectedGroupDiff.get().getOriginalGroupRoot(), groupDiff.get().getOriginalGroupRoot()); diff --git a/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java new file mode 100644 index 00000000000..1b63e26cbb3 --- /dev/null +++ b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java @@ -0,0 +1,666 @@ +package org.jabref.logic.bst; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.jabref.logic.bst.util.BstCaseChangersTest; +import org.jabref.logic.bst.util.BstNameFormatterTest; +import org.jabref.logic.bst.util.BstPurifierTest; +import org.jabref.logic.bst.util.BstTextPrefixerTest; +import org.jabref.logic.bst.util.BstWidthCalculatorTest; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.types.StandardEntryType; + +import org.antlr.v4.runtime.RecognitionException; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +/** + * For additional tests see for + * + * purify: {@link BstPurifierTest} + * width: {@link BstWidthCalculatorTest} + * format.name: {@link BstNameFormatterTest} + * change.case: {@link BstCaseChangersTest} + * prefix: {@link BstTextPrefixerTest} + * + */ +class BstFunctionsTest { + @Test + public void testCompareFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test.compare } { + #5 #5 = % TRUE + #1 #2 = % FALSE + #3 #4 < % TRUE + #4 #3 < % FALSE + #4 #4 < % FALSE + #3 #4 > % FALSE + #4 #3 > % TRUE + #4 #4 > % FALSE + "H" "H" = % TRUE + "H" "Ha" = % FALSE + } + EXECUTE { test.compare } + """); + + vm.render(Collections.emptyList()); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 #1 + % 2 + #5 #2 - % 3 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(3, vm.getStack().pop()); + assertEquals(2, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctionTypeMismatch() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 "HELLO" + % Should throw exception + } + EXECUTE { test } + """); + + assertThrows(BstVMException.class, () -> vm.render(Collections.emptyList())); + } + + @Test + public void testStringOperations() throws RecognitionException { + // Test for concat (*) and add.period + BstVM vm = new BstVM(""" + FUNCTION { test } { + "H" "ello" * % Hello + "Johnny" add.period$ % Johnny. + "Johnny." add.period$ % Johnny. + "Johnny!" add.period$ % Johnny! + "Johnny?" add.period$ % Johnny? + "Johnny} }}}" add.period$ % Johnny.} + "Johnny!}" add.period$ % Johnny!} + "Johnny?}" add.period$ % Johnny?} + "Johnny.}" add.period$ % Johnny.} + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?}", vm.getStack().pop()); + assertEquals("Johnny!}", vm.getStack().pop()); + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?", vm.getStack().pop()); + assertEquals("Johnny!", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Hello", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testMissing() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { test } { title missing$ cite$ } + ITERATE { test } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Article) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "No title")); + + vm.render(testEntries); + + assertEquals("test", vm.getStack().pop()); // cite + assertEquals(BstVM.TRUE, vm.getStack().pop()); // missing title + assertEquals("canh05", vm.getStack().pop()); // cite + assertEquals(BstVM.FALSE, vm.getStack().pop()); // missing title + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testNumNames() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "Johnny Foo { and } Mary Bar" num.names$ + "Johnny Foo and Mary Bar" num.names$ + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(2, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSubstring() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "123456789" #2 #1 substring$ % 2 + "123456789" #4 global.max$ substring$ % 456789 + "123456789" #1 #9 substring$ % 123456789 + "123456789" #1 #10 substring$ % 123456789 + "123456789" #1 #99 substring$ % 123456789 + "123456789" #-7 #3 substring$ % 123 + "123456789" #-1 #1 substring$ % 9 + "123456789" #-1 #3 substring$ % 789 + "123456789" #-2 #2 substring$ % 78 + } EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("78", vm.getStack().pop()); + assertEquals("789", vm.getStack().pop()); + assertEquals("9", vm.getStack().pop()); + assertEquals("123", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("456789", vm.getStack().pop()); + assertEquals("2", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testEmpty() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + READ + STRINGS { s } + FUNCTION { test } { + s empty$ % TRUE + "" empty$ % TRUE + " " empty$ % TRUE + title empty$ % TRUE + " HALLO " empty$ % FALSE + } + ITERATE { test } + """); + List testEntry = List.of(new BibEntry(StandardEntryType.Article)); + + vm.render(testEntry); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameStatic() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { format }{ "Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin" #1 "{vv~}{ll}{, jj}{, f}?" format.name$ } + EXECUTE { format } + """); + List v = Collections.emptyList(); + + vm.render(v); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameInEntries() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { author } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { format }{ author #2 "{vv~}{ll}{, jj}{, f}?" format.name$ } + ITERATE { format } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "Jonathan Meyer and Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin")); + + vm.render(testEntries); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals("Annabi, H?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChangeCase() throws RecognitionException { + BstVM vm = new BstVM(""" + STRINGS { title } + READ + FUNCTION { format.title } { + duplicate$ empty$ + { pop$ "" } + { "t" change.case$ } + if$ + } + FUNCTION { test } { + "hello world" "u" change.case$ format.title + "Hello World" format.title + "" format.title + "{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase" "u" change.case$ format.title + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase", + vm.getStack().pop()); + assertEquals("", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testTextLength() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "hello world" text.length$ % 11 + "Hello {W}orld" text.length$ % 11 + "" text.length$ % 0 + "{A}{D}/{Cycle}" text.length$ % 8 + "{\\This is one character}" text.length$ % 1 + "{\\This {is} {one} {c{h}}aracter as well}" text.length$ % 1 + "{\\And this too" text.length$ % 1 + "These are {\\11}" text.length$ % 11 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(11, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(8, vm.getStack().pop()); + assertEquals(0, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testIntToStr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { #3 int.to.str$ #9999 int.to.str$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("9999", vm.getStack().pop()); + assertEquals("3", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToInt() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(72, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToIntIntToChr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ int.to.chr$ } + EXECUTE {test} + """); + + vm.render(Collections.emptyList()); + + assertEquals("H", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + SORT + FUNCTION { test } { type$ } + ITERATE { test } + """); + List testEntries = List.of( + new BibEntry(StandardEntryType.Article).withCitationKey("a"), + new BibEntry(StandardEntryType.Book).withCitationKey("b"), + new BibEntry(StandardEntryType.Misc).withCitationKey("c"), + new BibEntry(StandardEntryType.InProceedings).withCitationKey("d")); + + vm.render(testEntries); + + assertEquals("inproceedings", vm.getStack().pop()); + assertEquals("misc", vm.getStack().pop()); + assertEquals("book", vm.getStack().pop()); + assertEquals("article", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testCallType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { inproceedings }{ "InProceedings called on " title * } + FUNCTION { book }{ "Book called on " title * } + ITERATE { call.type$ } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.TITLE, "Test")); + + vm.render(testEntries); + + assertEquals("Book called on Test", vm.getStack().pop()); + assertEquals( + "InProceedings called on Effective work practices for floss development: A model and propositions", + vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSwap() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { a } { #3 "Hallo" swap$ } + EXECUTE { a } + """); + + List
- * We use null for the missing entry designator. - */ - private void read(BibDatabase bibDatabase) { - FieldWriter fieldWriter = new FieldWriter(new FieldWriterPreferences(true, List.of(StandardField.MONTH), new FieldContentFormatterPreferences())); - for (BstEntry e : entries) { - for (Map.Entry mEntry : e.fields.entrySet()) { - Field field = FieldFactory.parseField(mEntry.getKey()); - String fieldValue = e.entry.getResolvedFieldOrAlias(field, bibDatabase) - .map(content -> { - try { - String result = fieldWriter.write(field, content); - if (result.startsWith("{")) { - // Strip enclosing {} from the output - return result.substring(1, result.length() - 1); - } - if (field == StandardField.MONTH) { - // We don't have the internal BibTeX strings at hand. - // We nevertheless want to have the full month name. - // Thus, we lookup the full month name here. - return Month.parse(result) - .map(month -> month.getFullName()) - .orElse(result); - } - return result; - } catch (InvalidFieldValueException invalidFieldValueException) { - // in case there is something wrong with the content, just return the content itself - return content; - } - }) - .orElse(null); - mEntry.setValue(fieldValue); - } - } - - for (BstEntry e : entries) { - if (!e.fields.containsKey(StandardField.CROSSREF.getName())) { - e.fields.put(StandardField.CROSSREF.getName(), null); - } - } - } - - /** - * Defines a string macro. It has two arguments; the first is the macro's name, which is treated like any other - * variable or function name, and the second is its definition, which must be double-quote-delimited. You must have - * one for each three-letter month abbreviation; in addition, you should have one for common journal names. The - * user's database may override any definition you define using this command. If you want to define a string the - * user can't touch, use the FUNCTION command, which has a compatible syntax. - */ - private void macro(Tree child) { - String name = child.getChild(0).getText(); - String replacement = child.getChild(1).getText(); - functions.put(name, new MacroFunction(replacement)); - } - - public class MacroFunction implements BstFunction { - - private final String replacement; - - public MacroFunction(String replacement) { - this.replacement = replacement; - } - - @Override - public void execute(BstEntry context) { - VM.this.push(replacement); - } - } - - /** - * Declares the fields and entry variables. It has three arguments, each a (possibly empty) list of variable names. - * The three lists are of: fields, integer entry variables, and string entry variables. There is an additional field - * that BibTEX automatically declares, crossref, used for cross referencing. And there is an additional string entry - * variable automatically declared, sort.key$, used by the SORT command. Each of these variables has a value for - * each entry on the list. - */ - private void entry(Tree child) { - // Fields first - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - - for (BstEntry entry : entries) { - entry.fields.put(name, null); - } - } - - // Integers - t = child.getChild(1); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - - for (BstEntry entry : entries) { - entry.localIntegers.put(name, 0); - } - } - // Strings - t = child.getChild(2); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - for (BstEntry entry : entries) { - entry.localStrings.put(name, null); - } - } - for (BstEntry entry : entries) { - entry.localStrings.put("sort.key$", null); - } - } - - private void reverse(Tree child) { - BstFunction f = functions.get(child.getChild(0).getText()); - - ListIterator i = entries.listIterator(entries.size()); - while (i.hasPrevious()) { - f.execute(i.previous()); - } - } - - private void iterate(Tree child) { - BstFunction f = functions.get(child.getChild(0).getText()); - - for (BstEntry entry : entries) { - f.execute(entry); - } - } - - /** - * Sorts the entry list using the values of the string entry variable sort.key$. It has no arguments. - */ - private void sort() { - entries.sort(Comparator.comparing(o -> (o.localStrings.get("sort.key$")))); - } - - private void executeInContext(Object o, BstEntry context) { - if (o instanceof Tree) { - Tree t = (Tree) o; - new StackFunction(t).execute(context); - } else if (o instanceof Identifier) { - execute(((Identifier) o).getName(), context); - } - } - - private void execute(Tree child) { - execute(child.getChild(0).getText(), null); - } - - public class StackFunction implements BstFunction { - - private final Tree localTree; - - public StackFunction(Tree stack) { - localTree = stack; - } - - public Tree getTree() { - return localTree; - } - - @Override - public void execute(BstEntry context) { - for (int i = 0; i < localTree.getChildCount(); i++) { - Tree c = localTree.getChild(i); - try { - - switch (c.getType()) { - case BstParser.STRING: - String s = c.getText(); - push(s.substring(1, s.length() - 1)); - break; - case BstParser.INTEGER: - push(Integer.parseInt(c.getText().substring(1))); - break; - case BstParser.QUOTED: - push(new Identifier(c.getText().substring(1))); - break; - case BstParser.STACK: - push(c); - break; - default: - VM.this.execute(c.getText(), context); - break; - } - } catch (VMException e) { - if (file == null) { - LOGGER.error("ERROR " + e.getMessage() + " (" + c.getLine() + ")"); - } else { - LOGGER.error("ERROR " + e.getMessage() + " (" + file.getPath() + ":" - + c.getLine() + ")"); - } - throw e; - } - } - } - } - - private void push(Tree t) { - stack.push(t); - } - - private void execute(String name, BstEntry context) { - if (context != null) { - if (context.fields.containsKey(name)) { - stack.push(context.fields.get(name)); - return; - } - if (context.localStrings.containsKey(name)) { - stack.push(context.localStrings.get(name)); - return; - } - if (context.localIntegers.containsKey(name)) { - stack.push(context.localIntegers.get(name)); - return; - } - } - if (strings.containsKey(name)) { - stack.push(strings.get(name)); - return; - } - if (integers.containsKey(name)) { - stack.push(integers.get(name)); - return; - } - - if (functions.containsKey(name)) { - // OK to have a null context - functions.get(name).execute(context); - return; - } - - throw new VMException("No matching identifier found: " + name); - } - - private void function(Tree child) { - String name = child.getChild(0).getText(); - Tree localStack = child.getChild(1); - functions.put(name, new StackFunction(localStack)); - } - - /** - * Declares global integer variables. It has one argument, a list of variable names. There are two such - * automatically-declared variables, entry.max$ and global.max$, used for limiting the lengths of string vari- - * ables. You may have any number of these commands, but a variable's declaration must precede its use. - */ - private void integers(Tree child) { - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - integers.put(name, 0); - } - } - - /** - * Declares global string variables. It has one argument, a list of variable names. You may have any number of these - * commands, but a variable's declaration must precede its use. - * - * @param child - */ - private void strings(Tree child) { - Tree t = child.getChild(0); - - for (int i = 0; i < t.getChildCount(); i++) { - String name = t.getChild(i).getText(); - strings.put(name, null); - } - } - - public static class BstEntry { - - public final BibEntry entry; - - public final Map localStrings = new HashMap<>(); - - // keys filled by org.jabref.logic.bst.VM.entry based on the contents of the bst file - public final Map fields = new HashMap<>(); - - public final Map localIntegers = new HashMap<>(); - - public BstEntry(BibEntry e) { - this.entry = e; - } - } - - private void push(Integer integer) { - stack.push(integer); - } - - private void push(String string) { - stack.push(string); - } - - private void push(Identifier identifier) { - stack.push(identifier); - } - - public Map getStrings() { - return strings; - } - - public Map getIntegers() { - return integers; - } - - public List getEntries() { - return entries; - } - - public Map getFunctions() { - return functions; - } - - public Stack getStack() { - return stack; - } - - @Override - public void warn(String string) { - LOGGER.warn(string); - } -} diff --git a/src/main/java/org/jabref/logic/bst/VMException.java b/src/main/java/org/jabref/logic/bst/VMException.java deleted file mode 100644 index f46c5cd5277..00000000000 --- a/src/main/java/org/jabref/logic/bst/VMException.java +++ /dev/null @@ -1,8 +0,0 @@ -package org.jabref.logic.bst; - -public class VMException extends RuntimeException { - - public VMException(String string) { - super(string); - } -} diff --git a/src/main/java/org/jabref/logic/bst/Warn.java b/src/main/java/org/jabref/logic/bst/Warn.java deleted file mode 100644 index 7a524ad9834..00000000000 --- a/src/main/java/org/jabref/logic/bst/Warn.java +++ /dev/null @@ -1,7 +0,0 @@ -package org.jabref.logic.bst; - -@FunctionalInterface -public interface Warn { - - void warn(String s); -} diff --git a/src/main/java/org/jabref/logic/bst/WidthFunction.java b/src/main/java/org/jabref/logic/bst/WidthFunction.java deleted file mode 100644 index 06784267bf0..00000000000 --- a/src/main/java/org/jabref/logic/bst/WidthFunction.java +++ /dev/null @@ -1,43 +0,0 @@ -package org.jabref.logic.bst; - -import java.util.Stack; - -import org.jabref.logic.bst.VM.BstEntry; -import org.jabref.logic.bst.VM.BstFunction; - -/** - * The |built_in| function {\.{width\$}} pops the top (string) literal and - * pushes the integer that represents its width in units specified by the - * |char_width| array. This function takes the literal literally; that is, it - * assumes each character in the string is to be printed as is, regardless of - * whether the character has a special meaning to \TeX, except that special - * characters (even without their |right_brace|s) are handled specially. If the - * literal isn't a string, it complains and pushes~0. - * - */ -public class WidthFunction implements BstFunction { - - private final VM vm; - - public WidthFunction(VM vm) { - this.vm = vm; - } - - @Override - public void execute(BstEntry context) { - Stack stack = vm.getStack(); - - if (stack.isEmpty()) { - throw new VMException("Not enough operands on stack for operation width$"); - } - Object o1 = stack.pop(); - - if (!(o1 instanceof String)) { - vm.warn("A string is needed for change.case$"); - stack.push(0); - return; - } - - stack.push(BibtexWidth.width((String) o1)); - } -} diff --git a/src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java b/src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java similarity index 83% rename from src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java rename to src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java index e05ce998e80..dd3e7c7b05c 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexCaseChanger.java +++ b/src/main/java/org/jabref/logic/bst/util/BstCaseChanger.java @@ -1,4 +1,4 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; import java.util.Locale; import java.util.Optional; @@ -6,9 +6,9 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; -public final class BibtexCaseChanger { +public final class BstCaseChanger { - private static final Logger LOGGER = LoggerFactory.getLogger(BibtexCaseChanger.class); + private static final Logger LOGGER = LoggerFactory.getLogger(BstCaseChanger.class); // stores whether the char before the current char was a colon private boolean prevColon = true; @@ -16,7 +16,7 @@ public final class BibtexCaseChanger { // global variable to store the current brace level private int braceLevel; - public enum FORMAT_MODE { + public enum FormatMode { // First character and character after a ":" as upper case - everything else in lower case. Obey {}. TITLE_LOWERS('t'), @@ -40,7 +40,7 @@ public enum FORMAT_MODE { private final char asChar; - FORMAT_MODE(char asChar) { + FormatMode(char asChar) { this.asChar = asChar; } @@ -53,17 +53,21 @@ public char asChar() { * * @throws IllegalArgumentException if char is not 't', 'l', 'u' */ - public static FORMAT_MODE getFormatModeForBSTFormat(final char bstFormat) { - for (FORMAT_MODE mode : FORMAT_MODE.values()) { + public static FormatMode of(final char bstFormat) { + for (FormatMode mode : FormatMode.values()) { if (mode.asChar == bstFormat) { return mode; } } throw new IllegalArgumentException(); } + + public static FormatMode of(final String bstFormat) { + return of(bstFormat.toLowerCase(Locale.ROOT).charAt(0)); + } } - private BibtexCaseChanger() { + private BstCaseChanger() { } /** @@ -72,11 +76,11 @@ private BibtexCaseChanger() { * @param s the string to handle * @param format the format */ - public static String changeCase(String s, FORMAT_MODE format) { - return (new BibtexCaseChanger()).doChangeCase(s, format); + public static String changeCase(String s, FormatMode format) { + return (new BstCaseChanger()).doChangeCase(s, format); } - private String doChangeCase(String s, FORMAT_MODE format) { + private String doChangeCase(String s, FormatMode format) { char[] c = s.toCharArray(); StringBuilder sb = new StringBuilder(); @@ -93,7 +97,7 @@ private String doChangeCase(String s, FORMAT_MODE format) { i++; continue; } - if ((format == FORMAT_MODE.TITLE_LOWERS) && ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1])))) { + if ((format == FormatMode.TITLE_LOWERS) && ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1])))) { sb.append('{'); i++; prevColon = false; @@ -136,12 +140,9 @@ private String doChangeCase(String s, FORMAT_MODE format) { * is other stuff, too, between braces, but it doesn't try to do anything * special with |colon|s. * - * @param c * @param start the current position. It points to the opening brace - * @param format - * @return */ - private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MODE format) { + private int convertSpecialChar(StringBuilder sb, char[] c, int start, FormatMode format) { int i = start; sb.append(c[i]); @@ -152,7 +153,7 @@ private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MOD i++; // skip over the |backslash| - Optional s = BibtexCaseChanger.findSpecialChar(c, i); + Optional s = BstCaseChanger.findSpecialChar(c, i); if (s.isPresent()) { i = convertAccented(c, i, s.get(), sb, format); } @@ -174,14 +175,9 @@ private int convertSpecialChar(StringBuilder sb, char[] c, int start, FORMAT_MOD * up) and append the result to the stringBuffer, return the updated * position. * - * @param c - * @param start - * @param s - * @param sb - * @param format * @return the new position */ - private int convertAccented(char[] c, int start, String s, StringBuilder sb, FORMAT_MODE format) { + private int convertAccented(char[] c, int start, String s, StringBuilder sb, FormatMode format) { int pos = start; pos += s.length(); @@ -214,29 +210,27 @@ private int convertAccented(char[] c, int start, String s, StringBuilder sb, FOR return pos; } - private int convertNonControl(char[] c, int start, StringBuilder sb, FORMAT_MODE format) { + private int convertNonControl(char[] c, int start, StringBuilder sb, FormatMode format) { int pos = start; switch (format) { - case TITLE_LOWERS: - case ALL_LOWERS: + case TITLE_LOWERS, ALL_LOWERS -> { sb.append(Character.toLowerCase(c[pos])); pos++; - break; - case ALL_UPPERS: + } + case ALL_UPPERS -> { sb.append(Character.toUpperCase(c[pos])); pos++; - break; - default: - LOGGER.info("convertNonControl - Unknown format: " + format); - break; + } + default -> + LOGGER.info("convertNonControl - Unknown format: " + format); } return pos; } - private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, FORMAT_MODE format) { + private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, FormatMode format) { int i = start; switch (format) { - case TITLE_LOWERS: + case TITLE_LOWERS -> { if ((i == 0) || (prevColon && Character.isWhitespace(c[i - 1]))) { sb.append(c[i]); } else { @@ -247,16 +241,13 @@ private int convertCharIfBraceLevelIsZero(char[] c, int start, StringBuilder sb, } else if (!Character.isWhitespace(c[i])) { prevColon = false; } - break; - case ALL_LOWERS: - sb.append(Character.toLowerCase(c[i])); - break; - case ALL_UPPERS: - sb.append(Character.toUpperCase(c[i])); - break; - default: - LOGGER.info("convertCharIfBraceLevelIsZero - Unknown format: " + format); - break; + } + case ALL_LOWERS -> + sb.append(Character.toLowerCase(c[i])); + case ALL_UPPERS -> + sb.append(Character.toUpperCase(c[i])); + default -> + LOGGER.info("convertCharIfBraceLevelIsZero - Unknown format: " + format); } i++; return i; diff --git a/src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java b/src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java similarity index 76% rename from src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java rename to src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java index aecf571bdf4..c4986dc12dd 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexNameFormatter.java +++ b/src/main/java/org/jabref/logic/bst/util/BstNameFormatter.java @@ -1,13 +1,17 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; import java.util.Arrays; import java.util.Locale; import java.util.Optional; import java.util.stream.Collectors; +import org.jabref.logic.bst.BstVMException; import org.jabref.model.entry.Author; import org.jabref.model.entry.AuthorList; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + /** * From Bibtex: * @@ -24,9 +28,10 @@ * Sounds easy - is a nightmare... X-( * */ -public class BibtexNameFormatter { +public class BstNameFormatter { + private static final Logger LOGGER = LoggerFactory.getLogger(BstNameFormatter.class); - private BibtexNameFormatter() { + private BstNameFormatter() { } /** @@ -35,23 +40,18 @@ private BibtexNameFormatter() { * @param authorsNameList The string from an author field * @param whichName index of the list, starting with 1 * @param formatString TODO - * @param warn collects the warnings, may-be-null - * @return */ - public static String formatName(String authorsNameList, int whichName, String formatString, Warn warn) { + public static String formatName(String authorsNameList, int whichName, String formatString) { AuthorList al = AuthorList.parse(authorsNameList); if ((whichName < 1) && (whichName > al.getNumberOfAuthors())) { - warn.warn("AuthorList " + authorsNameList + " does not contain an author with number " + whichName); + LOGGER.warn("AuthorList {} does not contain an author with number {}", authorsNameList, whichName); return ""; } - return BibtexNameFormatter.formatName(al.getAuthor(whichName - 1), formatString, warn); + return BstNameFormatter.formatName(al.getAuthor(whichName - 1), formatString); } - /** - * @param warn collects the warnings, may-be-null - */ - public static String formatName(Author author, String format, Warn warn) { + public static String formatName(Author author, String format) { StringBuilder sb = new StringBuilder(); char[] c = format.toCharArray(); @@ -81,11 +81,7 @@ public static String formatName(Author author, String format, Warn warn) { } if ((braceLevel == 1) && Character.isLetter(c[i])) { if ("fvlj".indexOf(c[i]) == -1) { - if (warn != null) { - warn.warn( - "Format string in format.name$ may only contain fvlj on brace level 1 in group " - + group + ": " + format); - } + LOGGER.warn("Format string in format.name$ may only contain fvlj on brace level 1 in group {}: {}", group, format); } else { level1Chars.append(c[i]); } @@ -99,31 +95,26 @@ public static String formatName(Author author, String format, Warn warn) { continue; } - if ((control.length() > 2) && (warn != null)) { - warn.warn("Format string in format.name$ may only be one or two character long on brace level 1 in group " + group + ": " + format); + if ((control.length() > 2)) { + LOGGER.warn("Format string in format.name$ may only be one or two character long on brace level 1 in group {}: {}", group, format); } char type = control.charAt(0); - Optional tokenS; - switch (type) { - case 'f': - tokenS = author.getFirst(); - break; - case 'v': - tokenS = author.getVon(); - break; - case 'l': - tokenS = author.getLast(); - break; - case 'j': - tokenS = author.getJr(); - break; - default: - throw new VMException("Internal error"); - } - - if (!tokenS.isPresent()) { + Optional tokenS = switch (type) { + case 'f' -> + author.getFirst(); + case 'v' -> + author.getVon(); + case 'l' -> + author.getLast(); + case 'j' -> + author.getJr(); + default -> + throw new BstVMException("Internal error"); + }; + + if (tokenS.isEmpty()) { i++; continue; } @@ -135,9 +126,7 @@ public static String formatName(Author author, String format, Warn warn) { if (control.charAt(1) == control.charAt(0)) { abbreviateThatIsSingleLetter = false; } else { - if (warn != null) { - warn.warn("Format string in format.name$ may only contain one type of vlfj on brace level 1 in group " + group + ": " + format); - } + LOGGER.warn("Format string in format.name$ may only contain one type of vlfj on brace level 1 in group {}: {}", group, format); } } @@ -162,7 +151,7 @@ public static String formatName(Author author, String format, Warn warn) { } if (((j + 1) < d.length) && (d[j + 1] == '{')) { StringBuilder interTokenSb = new StringBuilder(); - j = BibtexNameFormatter.consumeToMatchingBrace(interTokenSb, d, j + 1); + j = BstNameFormatter.consumeToMatchingBrace(interTokenSb, d, j + 1); interToken = interTokenSb.substring(1, interTokenSb.length() - 1); } @@ -171,7 +160,7 @@ public static String formatName(Author author, String format, Warn warn) { if (abbreviateThatIsSingleLetter) { String[] dashes = token.split("-"); - token = Arrays.asList(dashes).stream().map(BibtexNameFormatter::getFirstCharOfString) + token = Arrays.stream(dashes).map(BstNameFormatter::getFirstCharOfString) .collect(Collectors.joining(".-")); } @@ -187,7 +176,7 @@ public static String formatName(Author author, String format, Warn warn) { // No clue what this means (What the hell are tokens anyway??? // if (lex_class[name_sep_char[cur_token]] = sep_char) then // append_ex_buf_char_and_check (name_sep_char[cur_token]) - if ((k == (tokens.length - 2)) || (BibtexNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 3) < 3)) { + if ((k == (tokens.length - 2)) || (BstNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 3) < 3)) { sb.append('~'); } else { sb.append(' '); @@ -212,7 +201,7 @@ public static String formatName(Author author, String format, Warn warn) { if (sb.length() > 0) { boolean noDisTie = false; if ((sb.charAt(sb.length() - 1) == '~') && - ((BibtexNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 4) >= 4) || + ((BstNameFormatter.numberOfChars(sb.substring(groupStart, sb.length()), 4) >= 4) || ((sb.length() > 1) && (noDisTie = sb.charAt(sb.length() - 2) == '~')))) { sb.deleteCharAt(sb.length() - 1); if (!noDisTie) { @@ -221,16 +210,14 @@ public static String formatName(Author author, String format, Warn warn) { } } } else if (c[i] == '}') { - if (warn != null) { - warn.warn("Unmatched brace in format string: " + format); - } + LOGGER.warn("Unmatched brace in format string: {}", format); } else { sb.append(c[i]); // verbatim } i++; } - if ((braceLevel != 0) && (warn != null)) { - warn.warn("Unbalanced brace in format string for nameFormat: " + format); + if ((braceLevel != 0)) { + LOGGER.warn("Unbalanced brace in format string for nameFormat: {}", format); } return sb.toString(); @@ -268,7 +255,7 @@ public static String getFirstCharOfString(String s) { } if ((c[i] == '{') && ((i + 1) < c.length) && (c[i + 1] == '\\')) { StringBuilder sb = new StringBuilder(); - BibtexNameFormatter.consumeToMatchingBrace(sb, c, i); + BstNameFormatter.consumeToMatchingBrace(sb, c, i); return sb.toString(); } } diff --git a/src/main/java/org/jabref/logic/bst/BibtexPurify.java b/src/main/java/org/jabref/logic/bst/util/BstPurifier.java similarity index 77% rename from src/main/java/org/jabref/logic/bst/BibtexPurify.java rename to src/main/java/org/jabref/logic/bst/util/BstPurifier.java index 08cb14db207..90818a0749f 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexPurify.java +++ b/src/main/java/org/jabref/logic/bst/util/BstPurifier.java @@ -1,4 +1,7 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * @@ -10,17 +13,13 @@ * pushes the null string. * */ -public class BibtexPurify { +public class BstPurifier { + private static final Logger LOGGER = LoggerFactory.getLogger(BstPurifier.class); - private BibtexPurify() { + private BstPurifier() { } - /** - * @param toPurify - * @param warn may-be-null - * @return - */ - public static String purify(String toPurify, Warn warn) { + public static String purify(String toPurify) { StringBuilder sb = new StringBuilder(); char[] cs = toPurify.toCharArray(); @@ -41,7 +40,7 @@ public static String purify(String toPurify, Warn warn) { i++; // skip brace while ((i < n) && (braceLevel > 0)) { i++; // skip backslash - BibtexCaseChanger.findSpecialChar(cs, i).ifPresent(sb::append); + BstCaseChanger.findSpecialChar(cs, i).ifPresent(sb::append); while ((i < n) && Character.isLetter(cs[i])) { i++; @@ -63,15 +62,13 @@ public static String purify(String toPurify, Warn warn) { if (braceLevel > 0) { braceLevel--; } else { - if (warn != null) { - warn.warn("Unbalanced brace in string for purify$: " + toPurify); - } + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPurify); } } i++; } - if ((braceLevel != 0) && (warn != null)) { - warn.warn("Unbalanced brace in string for purify$: " + toPurify); + if ((braceLevel != 0)) { + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPurify); } return sb.toString(); diff --git a/src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java b/src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java similarity index 84% rename from src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java rename to src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java index 721838e29fe..31cbc71de9b 100644 --- a/src/main/java/org/jabref/logic/bst/BibtexTextPrefix.java +++ b/src/main/java/org/jabref/logic/bst/util/BstTextPrefixer.java @@ -1,4 +1,7 @@ -package org.jabref.logic.bst; +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * The |built_in| function {\.{text.prefix\$}} pops the top two literals (the @@ -14,15 +17,13 @@ * complains and pushes the null string. * */ -public class BibtexTextPrefix { +public class BstTextPrefixer { + private static final Logger LOGGER = LoggerFactory.getLogger(BstTextPrefixer.class); - private BibtexTextPrefix() { + private BstTextPrefixer() { } - /** - * @param warn may-be-null - */ - public static String textPrefix(int inNumOfChars, String toPrefix, Warn warn) { + public static String textPrefix(int inNumOfChars, String toPrefix) { int numOfChars = inNumOfChars; StringBuilder sb = new StringBuilder(); @@ -53,15 +54,13 @@ public static String textPrefix(int inNumOfChars, String toPrefix, Warn warn) { if (braceLevel > 0) { braceLevel--; } else { - if (warn != null) { - warn.warn("Unbalanced brace in string for purify$: " + toPrefix); - } + LOGGER.warn("Unbalanced brace in string for purify$: {}", toPrefix); } } else { numOfChars--; } } - sb.append(toPrefix.substring(0, i)); + sb.append(toPrefix, 0, i); while (braceLevel > 0) { sb.append('}'); braceLevel--; diff --git a/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java b/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java new file mode 100644 index 00000000000..c286fe1497b --- /dev/null +++ b/src/main/java/org/jabref/logic/bst/util/BstWidthCalculator.java @@ -0,0 +1,241 @@ +package org.jabref.logic.bst.util; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * + * The |built_in| function {\.{purify\$}} pops the top (string) literal, removes + * nonalphanumeric characters except for |white_space| and |sep_char| characters + * (these get converted to a |space|) and removes certain alphabetic characters + * contained in the control sequences associated with a special character, and + * pushes the resulting string. If the literal isn't a string, it complains and + * pushes the null string. + * + */ +public class BstWidthCalculator { + + private static final Logger LOGGER = LoggerFactory.getLogger(BstWidthCalculator.class); + + /* + * Quoted from Bibtex: + * + * Now we initialize the system-dependent |char_width| array, for which + * |space| is the only |white_space| character given a nonzero printing + * width. The widths here are taken from Stanford's June~'87 $cmr10$~font + * and represent hundredths of a point (rounded), but since they're used + * only for relative comparisons, the units have no meaning. + */ + + private static int[] widths; + + static { + if (BstWidthCalculator.widths == null) { + BstWidthCalculator.widths = new int[128]; + + for (int i = 0; i < 128; i++) { + BstWidthCalculator.widths[i] = 0; + } + BstWidthCalculator.widths[32] = 278; + BstWidthCalculator.widths[33] = 278; + BstWidthCalculator.widths[34] = 500; + BstWidthCalculator.widths[35] = 833; + BstWidthCalculator.widths[36] = 500; + BstWidthCalculator.widths[37] = 833; + BstWidthCalculator.widths[38] = 778; + BstWidthCalculator.widths[39] = 278; + BstWidthCalculator.widths[40] = 389; + BstWidthCalculator.widths[41] = 389; + BstWidthCalculator.widths[42] = 500; + BstWidthCalculator.widths[43] = 778; + BstWidthCalculator.widths[44] = 278; + BstWidthCalculator.widths[45] = 333; + BstWidthCalculator.widths[46] = 278; + BstWidthCalculator.widths[47] = 500; + BstWidthCalculator.widths[48] = 500; + BstWidthCalculator.widths[49] = 500; + BstWidthCalculator.widths[50] = 500; + BstWidthCalculator.widths[51] = 500; + BstWidthCalculator.widths[52] = 500; + BstWidthCalculator.widths[53] = 500; + BstWidthCalculator.widths[54] = 500; + BstWidthCalculator.widths[55] = 500; + BstWidthCalculator.widths[56] = 500; + BstWidthCalculator.widths[57] = 500; + BstWidthCalculator.widths[58] = 278; + BstWidthCalculator.widths[59] = 278; + BstWidthCalculator.widths[60] = 278; + BstWidthCalculator.widths[61] = 778; + BstWidthCalculator.widths[62] = 472; + BstWidthCalculator.widths[63] = 472; + BstWidthCalculator.widths[64] = 778; + BstWidthCalculator.widths[65] = 750; + BstWidthCalculator.widths[66] = 708; + BstWidthCalculator.widths[67] = 722; + BstWidthCalculator.widths[68] = 764; + BstWidthCalculator.widths[69] = 681; + BstWidthCalculator.widths[70] = 653; + BstWidthCalculator.widths[71] = 785; + BstWidthCalculator.widths[72] = 750; + BstWidthCalculator.widths[73] = 361; + BstWidthCalculator.widths[74] = 514; + BstWidthCalculator.widths[75] = 778; + BstWidthCalculator.widths[76] = 625; + BstWidthCalculator.widths[77] = 917; + BstWidthCalculator.widths[78] = 750; + BstWidthCalculator.widths[79] = 778; + BstWidthCalculator.widths[80] = 681; + BstWidthCalculator.widths[81] = 778; + BstWidthCalculator.widths[82] = 736; + BstWidthCalculator.widths[83] = 556; + BstWidthCalculator.widths[84] = 722; + BstWidthCalculator.widths[85] = 750; + BstWidthCalculator.widths[86] = 750; + BstWidthCalculator.widths[87] = 1028; + BstWidthCalculator.widths[88] = 750; + BstWidthCalculator.widths[89] = 750; + BstWidthCalculator.widths[90] = 611; + BstWidthCalculator.widths[91] = 278; + BstWidthCalculator.widths[92] = 500; + BstWidthCalculator.widths[93] = 278; + BstWidthCalculator.widths[94] = 500; + BstWidthCalculator.widths[95] = 278; + BstWidthCalculator.widths[96] = 278; + BstWidthCalculator.widths[97] = 500; + BstWidthCalculator.widths[98] = 556; + BstWidthCalculator.widths[99] = 444; + BstWidthCalculator.widths[100] = 556; + BstWidthCalculator.widths[101] = 444; + BstWidthCalculator.widths[102] = 306; + BstWidthCalculator.widths[103] = 500; + BstWidthCalculator.widths[104] = 556; + BstWidthCalculator.widths[105] = 278; + BstWidthCalculator.widths[106] = 306; + BstWidthCalculator.widths[107] = 528; + BstWidthCalculator.widths[108] = 278; + BstWidthCalculator.widths[109] = 833; + BstWidthCalculator.widths[110] = 556; + BstWidthCalculator.widths[111] = 500; + BstWidthCalculator.widths[112] = 556; + BstWidthCalculator.widths[113] = 528; + BstWidthCalculator.widths[114] = 392; + BstWidthCalculator.widths[115] = 394; + BstWidthCalculator.widths[116] = 389; + BstWidthCalculator.widths[117] = 556; + BstWidthCalculator.widths[118] = 528; + BstWidthCalculator.widths[119] = 722; + BstWidthCalculator.widths[120] = 528; + BstWidthCalculator.widths[121] = 528; + BstWidthCalculator.widths[122] = 444; + BstWidthCalculator.widths[123] = 500; + BstWidthCalculator.widths[124] = 1000; + BstWidthCalculator.widths[125] = 500; + BstWidthCalculator.widths[126] = 500; + } + } + + private BstWidthCalculator() { + } + + private static int getSpecialCharWidth(char[] c, int pos) { + if ((pos + 1) < c.length) { + if ((c[pos] == 'o') && (c[pos + 1] == 'e')) { + return 778; + } + if ((c[pos] == 'O') && (c[pos + 1] == 'E')) { + return 1014; + } + if ((c[pos] == 'a') && (c[pos + 1] == 'e')) { + return 722; + } + if ((c[pos] == 'A') && (c[pos + 1] == 'E')) { + return 903; + } + if ((c[pos] == 's') && (c[pos + 1] == 's')) { + return 500; + } + } + return BstWidthCalculator.getCharWidth(c[pos]); + } + + public static int getCharWidth(char c) { + if ((c >= 0) && (c < 128)) { + return BstWidthCalculator.widths[c]; + } else { + return 0; + } + } + + public static int width(String toMeasure) { + /* + * From Bibtex: We use the natural width for all but special characters, + * and we complain if the string isn't brace-balanced. + */ + + int i = 0; + int n = toMeasure.length(); + int braceLevel = 0; + char[] c = toMeasure.toCharArray(); + int result = 0; + + /* + * From Bibtex: + * + * We use the natural widths of all characters except that some + * characters have no width: braces, control sequences (except for the + * usual 13 accented and foreign characters, whose widths are given in + * the next module), and |white_space| following control sequences (even + * a null control sequence). + * + */ + while (i < n) { + if (c[i] == '{') { + braceLevel++; + if ((braceLevel == 1) && ((i + 1) < n) && (c[i + 1] == '\\')) { + i++; // skip brace + while ((i < n) && (braceLevel > 0)) { + i++; // skip backslash + + int afterBackslash = i; + while ((i < n) && Character.isLetter(c[i])) { + i++; + } + if ((i < n) && (i == afterBackslash)) { + i++; // Skip non-alpha control seq + } else { + if (BstCaseChanger.findSpecialChar(c, afterBackslash).isPresent()) { + result += BstWidthCalculator.getSpecialCharWidth(c, afterBackslash); + } + } + while ((i < n) && Character.isWhitespace(c[i])) { + i++; + } + while ((i < n) && (braceLevel > 0) && (c[i] != '\\')) { + if (c[i] == '}') { + braceLevel--; + } else if (c[i] == '{') { + braceLevel++; + } else { + result += BstWidthCalculator.getCharWidth(c[i]); + } + i++; + } + } + continue; + } + } else if (c[i] == '}') { + if (braceLevel > 0) { + braceLevel--; + } else { + LOGGER.warn("Too many closing braces in string: " + toMeasure); + } + } + result += BstWidthCalculator.getCharWidth(c[i]); + i++; + } + if (braceLevel > 0) { + LOGGER.warn("No enough closing braces in string: " + toMeasure); + } + return result; + } +} diff --git a/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java b/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java index c0f564800c0..1b98ae4aa15 100644 --- a/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java +++ b/src/main/java/org/jabref/logic/exporter/BibtexDatabaseWriter.java @@ -110,7 +110,11 @@ protected void writeEntryTypeDefinition(BibEntryType customType) throws IOExcept @Override protected void writeProlog(BibDatabaseContext bibDatabaseContext, Charset encoding) throws IOException { - if ((encoding == null) || (encoding == StandardCharsets.UTF_8)) { + // We write the encoding if + // - it is provided (!= null) + // - explicitly set in the .bib file OR not equal to UTF_8 + // Otherwise, we do not write anything and return + if ((encoding == null) || (!bibDatabaseContext.getMetaData().getEncodingExplicitlySupplied() && (encoding == StandardCharsets.UTF_8))) { return; } diff --git a/src/main/java/org/jabref/logic/importer/ImportFormatReader.java b/src/main/java/org/jabref/logic/importer/ImportFormatReader.java index 617cd5c0dba..f3b8826171d 100644 --- a/src/main/java/org/jabref/logic/importer/ImportFormatReader.java +++ b/src/main/java/org/jabref/logic/importer/ImportFormatReader.java @@ -195,7 +195,7 @@ public UnknownFormatImport importUnknownFormat(Path filePath, FileUpdateMonitor parserResult.setPath(filePath); return new UnknownFormatImport(ImportFormatReader.BIBTEX_FORMAT, parserResult); } else { - throw new ImportException(Localization.lang("Could not find a suitable import format.")); + throw new ImportException(parserResult.getErrorMessage()); } } catch (IOException ignore) { // Ignored diff --git a/src/main/java/org/jabref/logic/importer/Importer.java b/src/main/java/org/jabref/logic/importer/Importer.java index f6ffc966214..56bde6f0e3e 100644 --- a/src/main/java/org/jabref/logic/importer/Importer.java +++ b/src/main/java/org/jabref/logic/importer/Importer.java @@ -123,12 +123,10 @@ protected static Charset getCharset(BufferedInputStream bufferedInputStream) { if ((matches == null) || (matches.length == 0)) { return defaultCharSet; } - if (Arrays.stream(matches).anyMatch(singleCharset -> singleCharset.getName().equals(defaultCharSet.toString()))) { + + if (Arrays.stream(matches).anyMatch(charset -> "ASCII".equals(charset.getName()))) { return defaultCharSet; } - if (Arrays.stream(matches).anyMatch(singleCharset -> singleCharset.getName().equals(StandardCharsets.UTF_16.toString()))) { - return StandardCharsets.UTF_16; - } if (matches[0] != null) { return Charset.forName(matches[0].getName()); @@ -166,7 +164,7 @@ public static BufferedReader getReader(Path filePath) throws IOException { return new BufferedReader(new InputStreamReader(stream, StandardCharsets.UTF_8)); } - public static BufferedReader getReader(InputStream stream) throws IOException { + public static BufferedReader getReader(InputStream stream) { BufferedInputStream bufferedInputStream = new BufferedInputStream(stream); Charset charset = getCharset(bufferedInputStream); InputStreamReader reader = new InputStreamReader(bufferedInputStream, charset); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java index 5fb3d8ed900..65301e2a7a1 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibTeXMLImporter.java @@ -184,7 +184,7 @@ private void parse(T entryType, Map fields) { } else if (isMethodToIgnore(method.getName())) { continue; } else if (method.getName().startsWith("get")) { - putIfValueNotNull(fields, FieldFactory.parseField(method.getName().replace("get", "")), (String) method.invoke(entryType)); + putIfValueNotNull(fields, FieldFactory.parseField(entryType, method.getName().replace("get", "")), (String) method.invoke(entryType)); } } catch (IllegalArgumentException | InvocationTargetException | IllegalAccessException e) { LOGGER.error("Could not invoke method", e); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java index e35fe6c73ad..590ae78de4e 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibtexImporter.java @@ -4,7 +4,10 @@ import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; +import java.io.InputStreamReader; import java.nio.charset.Charset; +import java.nio.charset.CharsetDecoder; +import java.nio.charset.CodingErrorAction; import java.nio.file.Files; import java.nio.file.Path; import java.util.Objects; @@ -68,18 +71,28 @@ public ParserResult importDatabase(Path filePath) throws IOException { } Charset encoding; + boolean encodingExplicitlySupplied; try (BufferedReader reader = Files.newBufferedReader(filePath, detectedCharset)) { Optional suppliedEncoding = getSuppliedEncoding(reader); LOGGER.debug("Supplied encoding: {}", suppliedEncoding); + encodingExplicitlySupplied = suppliedEncoding.isPresent(); // in case no encoding information is present, use the detected one encoding = suppliedEncoding.orElse(detectedCharset); LOGGER.debug("Encoding used to read the file: {}", encoding); } - try (BufferedReader reader = Files.newBufferedReader(filePath, encoding)) { + // We replace unreadable characters + // Unfortunately, no warning will be issued to the user + // As this is a very seldom case, we accept that + CharsetDecoder decoder = encoding.newDecoder(); + decoder.onMalformedInput(CodingErrorAction.REPLACE); + + try (InputStreamReader inputStreamReader = new InputStreamReader(Files.newInputStream(filePath), decoder); + BufferedReader reader = new BufferedReader(inputStreamReader)) { ParserResult parserResult = this.importDatabase(reader); parserResult.getMetaData().setEncoding(encoding); + parserResult.getMetaData().setEncodingExplicitlySupplied(encodingExplicitlySupplied); parserResult.setPath(filePath); if (parserResult.getMetaData().getMode().isEmpty()) { parserResult.getMetaData().setMode(BibDatabaseModeDetection.inferMode(parserResult.getDatabase())); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java b/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java index 3fe4fb684d6..666ae01a7fb 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/BibtexParser.java @@ -283,7 +283,7 @@ private void parseAndAddEntry(String type) { LOGGER.debug("Could not parse entry", ex); parserResult.addWarning(Localization.lang("Error occurred when parsing entry") + ": '" + ex.getMessage() - + "'. " + Localization.lang("Skipped entry.")); + + "'. " + "\n\n" + Localization.lang("JabRef skipped the entry.")); } } @@ -387,8 +387,8 @@ private String purge(String context, String stringToPurge) { } // strip empty lines while ((runningIndex < indexOfAt) && - (context.charAt(runningIndex) == '\r' || - context.charAt(runningIndex) == '\n')) { + ((context.charAt(runningIndex) == '\r') || + (context.charAt(runningIndex) == '\n'))) { runningIndex++; } return context.substring(runningIndex); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java index 2ff451e6dca..afbe5af657d 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/CffImporter.java @@ -14,6 +14,7 @@ import org.jabref.model.entry.Author; import org.jabref.model.entry.AuthorList; import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.BiblatexSoftwareField; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.StandardField; import org.jabref.model.entry.field.UnknownField; @@ -95,7 +96,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { StandardEntryType entryType = StandardEntryType.Software; // Map CFF fields to JabRef Fields - HashMap fieldMap = getFieldMappings(); + HashMap fieldMap = getFieldMappings(); for (Map.Entry property : citation.values.entrySet()) { if (fieldMap.containsKey(property.getKey())) { entryMap.put(fieldMap.get(property.getKey()), property.getValue()); @@ -120,7 +121,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { entryMap.put(StandardField.AUTHOR, authorStr); // Select DOI to keep - if (entryMap.get(StandardField.DOI) == null && citation.ids != null) { + if ((entryMap.get(StandardField.DOI) == null) && (citation.ids != null)) { List doiIds = citation.ids.stream() .filter(id -> id.type.equals("doi")) .collect(Collectors.toList()); @@ -137,14 +138,14 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { .collect(Collectors.toList()); if (swhIds.size() == 1) { - entryMap.put(StandardField.SWHID, swhIds.get(0)); + entryMap.put(BiblatexSoftwareField.SWHID, swhIds.get(0)); } else if (swhIds.size() > 1) { List relSwhIds = swhIds.stream() .filter(id -> id.split(":").length > 3) // quick filter for invalid swhids .filter(id -> id.split(":")[2].equals("rel")) .collect(Collectors.toList()); if (relSwhIds.size() == 1) { - entryMap.put(StandardField.SWHID, relSwhIds.get(0)); + entryMap.put(BiblatexSoftwareField.SWHID, relSwhIds.get(0)); } } } @@ -166,19 +167,19 @@ public boolean isRecognizedFormat(BufferedReader reader) throws IOException { try { citation = mapper.readValue(reader, CffFormat.class); - return citation != null && citation.values.get("title") != null; + return (citation != null) && (citation.values.get("title") != null); } catch (IOException e) { return false; } } - private HashMap getFieldMappings() { - HashMap fieldMappings = new HashMap<>(); + private HashMap getFieldMappings() { + HashMap fieldMappings = new HashMap<>(); fieldMappings.put("title", StandardField.TITLE); fieldMappings.put("version", StandardField.VERSION); fieldMappings.put("doi", StandardField.DOI); - fieldMappings.put("license", StandardField.LICENSE); - fieldMappings.put("repository", StandardField.REPOSITORY); + fieldMappings.put("license", BiblatexSoftwareField.LICENSE); + fieldMappings.put("repository", BiblatexSoftwareField.REPOSITORY); fieldMappings.put("url", StandardField.URL); fieldMappings.put("abstract", StandardField.ABSTRACT); fieldMappings.put("message", StandardField.COMMENT); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java index ab857becd0a..ab638ae4cf3 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/CopacImporter.java @@ -137,7 +137,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { } else if ("DT- ".equals(code)) { setOrAppend(b, new UnknownField("documenttype"), line.substring(4).trim(), ", "); } else { - setOrAppend(b, FieldFactory.parseField(code.substring(0, 2)), line.substring(4).trim(), ", "); + setOrAppend(b, FieldFactory.parseField(StandardEntryType.Book, line.substring(0, 2)), line.substring(4).trim(), ", "); } } results.add(b); diff --git a/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java index b502977f716..2845756b770 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/IsiImporter.java @@ -281,7 +281,7 @@ public ParserResult importDatabase(BufferedReader reader) throws IOException { if ("ER".equals(beg) || "EF".equals(beg) || "VR".equals(beg) || "FN".equals(beg)) { continue; } - hm.put(FieldFactory.parseField(beg), value); + hm.put(FieldFactory.parseField(type, beg), value); } } diff --git a/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java b/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java index 392fad9772b..b9e9eac39d4 100644 --- a/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java +++ b/src/main/java/org/jabref/logic/importer/fileformat/MedlineImporter.java @@ -425,7 +425,7 @@ private void addArticleIdList(Map fields, ArticleIdList articleId if ("pubmed".equals(id.getIdType())) { fields.put(StandardField.PMID, id.getContent()); } else { - fields.put(FieldFactory.parseField(id.getIdType()), id.getContent()); + fields.put(FieldFactory.parseField(StandardEntryType.Article, id.getIdType()), id.getContent()); } } } @@ -499,7 +499,7 @@ private void addKeyWords(Map fields, List allKeyword private void addOtherId(Map fields, List otherID) { for (OtherID id : otherID) { if ((id.getSource() != null) && (id.getContent() != null)) { - fields.put(FieldFactory.parseField(id.getSource()), id.getContent()); + fields.put(FieldFactory.parseField(StandardEntryType.Article, id.getSource()), id.getContent()); } } } diff --git a/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java b/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java index eb1e905adca..5f3d5e29c52 100644 --- a/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java +++ b/src/main/java/org/jabref/logic/importer/util/MetaDataParser.java @@ -65,6 +65,7 @@ public MetaData parse(MetaData metaData, Map data, Character key String user = entry.getKey().substring(MetaData.FILE_DIRECTORY.length() + 1); metaData.setUserFileDirectory(user, getSingleItem(value)); } else if (entry.getKey().startsWith(MetaData.SELECTOR_META_PREFIX)) { + // edge case, it might be one special field e.g. article from biblatex-apa, but we can't distinguish this from any other field and rather prefer to handle it as UnknownField metaData.addContentSelector(ContentSelectors.parse(FieldFactory.parseField(entry.getKey().substring(MetaData.SELECTOR_META_PREFIX.length())), StringUtil.unquote(entry.getValue(), MetaData.ESCAPE_CHARACTER))); } else if (entry.getKey().startsWith(MetaData.FILE_DIRECTORY + "Latex-")) { // The user name comes directly after "FILE_DIRECTORYLatex-" diff --git a/src/main/java/org/jabref/logic/layout/format/NameFormatter.java b/src/main/java/org/jabref/logic/layout/format/NameFormatter.java index 919ff4df2d5..e8812235435 100644 --- a/src/main/java/org/jabref/logic/layout/format/NameFormatter.java +++ b/src/main/java/org/jabref/logic/layout/format/NameFormatter.java @@ -5,7 +5,7 @@ import java.util.Map; import java.util.Objects; -import org.jabref.logic.bst.BibtexNameFormatter; +import org.jabref.logic.bst.util.BstNameFormatter; import org.jabref.logic.layout.LayoutFormatter; import org.jabref.model.entry.AuthorList; @@ -86,7 +86,7 @@ private static String format(String toFormat, AuthorList al, String[] formats) { for (int i = 1; i <= al.getNumberOfAuthors(); i++) { for (int j = 1; j < formats.length; j += 2) { if ("*".equals(formats[j])) { - sb.append(BibtexNameFormatter.formatName(toFormat, i, formats[j + 1], null)); + sb.append(BstNameFormatter.formatName(toFormat, i, formats[j + 1])); break; } else { String[] range = formats[j].split("\\.\\."); @@ -112,7 +112,7 @@ private static String format(String toFormat, AuthorList al, String[] formats) { } if ((s <= i) && (i <= e)) { - sb.append(BibtexNameFormatter.formatName(toFormat, i, formats[j + 1], null)); + sb.append(BstNameFormatter.formatName(toFormat, i, formats[j + 1])); break; } } diff --git a/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java b/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java index 24e897ae926..8ebc6bddd04 100644 --- a/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java +++ b/src/main/java/org/jabref/logic/pdf/search/indexing/IndexingTaskManager.java @@ -2,6 +2,7 @@ import java.util.List; import java.util.Queue; +import java.util.Set; import java.util.concurrent.ConcurrentLinkedQueue; import org.jabref.gui.util.BackgroundTask; @@ -88,12 +89,17 @@ public void createIndex(PdfIndexer indexer) { enqueueTask(() -> indexer.createIndex()); } - public void addToIndex(PdfIndexer indexer, BibDatabaseContext databaseContext) { + public void updateIndex(PdfIndexer indexer, BibDatabaseContext databaseContext) { + Set pathsToRemove = indexer.getListOfFilePaths(); for (BibEntry entry : databaseContext.getEntries()) { for (LinkedFile file : entry.getFiles()) { enqueueTask(() -> indexer.addToIndex(entry, file, databaseContext)); + pathsToRemove.remove(file.getLink()); } } + for (String pathToRemove : pathsToRemove) { + enqueueTask(() -> indexer.removeFromIndex(pathToRemove)); + } } public void addToIndex(PdfIndexer indexer, BibEntry entry, BibDatabaseContext databaseContext) { @@ -108,7 +114,7 @@ public void addToIndex(PdfIndexer indexer, BibEntry entry, List link public void removeFromIndex(PdfIndexer indexer, BibEntry entry, List linkedFiles) { for (LinkedFile file : linkedFiles) { - enqueueTask(() -> indexer.removeFromIndex(entry, file)); + enqueueTask(() -> indexer.removeFromIndex(file.getLink())); } } diff --git a/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java b/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java index 3b91be44f71..975cf83b1ec 100644 --- a/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java +++ b/src/main/java/org/jabref/logic/pdf/search/indexing/PdfIndexer.java @@ -4,8 +4,10 @@ import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.attribute.BasicFileAttributes; +import java.util.HashSet; import java.util.List; import java.util.Optional; +import java.util.Set; import java.util.concurrent.TimeUnit; import org.jabref.gui.LibraryTab; @@ -25,6 +27,8 @@ import org.apache.lucene.index.IndexWriterConfig; import org.apache.lucene.index.Term; import org.apache.lucene.search.IndexSearcher; +import org.apache.lucene.search.MatchAllDocsQuery; +import org.apache.lucene.search.ScoreDoc; import org.apache.lucene.search.TermQuery; import org.apache.lucene.search.TopDocs; import org.apache.lucene.store.Directory; @@ -110,19 +114,16 @@ public void addToIndex(BibEntry entry, LinkedFile linkedFile, BibDatabaseContext } /** - * Removes a pdf file linked to one entry in the database from the index + * Removes a pdf file identified by its path from the index * - * @param entry the entry the file is linked to - * @param linkedFile the link to the file to be removed + * @param linkedFilePath the path to the file to be removed */ - public void removeFromIndex(BibEntry entry, LinkedFile linkedFile) { + public void removeFromIndex(String linkedFilePath) { try (IndexWriter indexWriter = new IndexWriter( directoryToIndex, new IndexWriterConfig( new EnglishStemAnalyzer()).setOpenMode(IndexWriterConfig.OpenMode.CREATE_OR_APPEND))) { - if (!entry.getFiles().isEmpty()) { - indexWriter.deleteDocuments(new Term(SearchFieldConstants.PATH, linkedFile.getLink())); - } + indexWriter.deleteDocuments(new Term(SearchFieldConstants.PATH, linkedFilePath)); indexWriter.commit(); } catch (IOException e) { LOGGER.warn("Could not initialize the IndexWriter!", e); @@ -145,7 +146,7 @@ public void removeFromIndex(BibEntry entry) { */ public void removeFromIndex(BibEntry entry, List linkedFiles) { for (LinkedFile linkedFile : linkedFiles) { - removeFromIndex(entry, linkedFile); + removeFromIndex(linkedFile.getLink()); } } @@ -224,4 +225,25 @@ private void writeToIndex(BibEntry entry, LinkedFile linkedFile) { LOGGER.warn("Could not add the document {} to the index!", linkedFile.getLink(), e); } } + + /** + * Lists the paths of all the files that are stored in the index + * + * @return all file paths + */ + public Set getListOfFilePaths() { + Set paths = new HashSet<>(); + try (IndexReader reader = DirectoryReader.open(directoryToIndex)) { + IndexSearcher searcher = new IndexSearcher(reader); + MatchAllDocsQuery query = new MatchAllDocsQuery(); + TopDocs allDocs = searcher.search(query, Integer.MAX_VALUE); + for (ScoreDoc scoreDoc : allDocs.scoreDocs) { + Document doc = reader.document(scoreDoc.doc); + paths.add(doc.getField(SearchFieldConstants.PATH).stringValue()); + } + } catch (IOException e) { + return paths; + } + return paths; + } } diff --git a/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java b/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java index 260ec297c24..4b1a14fe1c9 100644 --- a/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java +++ b/src/main/java/org/jabref/logic/shared/prefs/SharedDatabasePreferences.java @@ -25,6 +25,8 @@ public class SharedDatabasePreferences { private static final String SHARED_DATABASE_NAME = "sharedDatabaseName"; private static final String SHARED_DATABASE_USER = "sharedDatabaseUser"; private static final String SHARED_DATABASE_PASSWORD = "sharedDatabasePassword"; + private static final String SHARED_DATABASE_FOLDER = "sharedDatabaseFolder"; + private static final String SHARED_DATABASE_AUTOSAVE = "sharedDatabaseAutosave"; private static final String SHARED_DATABASE_REMEMBER_PASSWORD = "sharedDatabaseRememberPassword"; private static final String SHARED_DATABASE_USE_SSL = "sharedDatabaseUseSSL"; private static final String SHARED_DATABASE_KEYSTORE_FILE = "sharedDatabaseKeyStoreFile"; @@ -77,6 +79,14 @@ public boolean getRememberPassword() { return internalPrefs.getBoolean(SHARED_DATABASE_REMEMBER_PASSWORD, false); } + public Optional getFolder() { + return getOptionalValue(SHARED_DATABASE_FOLDER); + } + + public boolean getAutosave() { + return internalPrefs.getBoolean(SHARED_DATABASE_AUTOSAVE, false); + } + public boolean isUseSSL() { return internalPrefs.getBoolean(SHARED_DATABASE_USE_SSL, false); } @@ -109,6 +119,14 @@ public void setRememberPassword(boolean rememberPassword) { internalPrefs.putBoolean(SHARED_DATABASE_REMEMBER_PASSWORD, rememberPassword); } + public void setFolder(String folder) { + internalPrefs.put(SHARED_DATABASE_FOLDER, folder); + } + + public void setAutosave(boolean autosave) { + internalPrefs.putBoolean(SHARED_DATABASE_AUTOSAVE, autosave); + } + public void setUseSSL(boolean useSSL) { internalPrefs.putBoolean(SHARED_DATABASE_USE_SSL, useSSL); } diff --git a/src/main/java/org/jabref/model/database/BibDatabase.java b/src/main/java/org/jabref/model/database/BibDatabase.java index 72a559ff663..52e4e841ce9 100644 --- a/src/main/java/org/jabref/model/database/BibDatabase.java +++ b/src/main/java/org/jabref/model/database/BibDatabase.java @@ -20,6 +20,7 @@ import javafx.collections.FXCollections; import javafx.collections.ObservableList; +import javafx.collections.ObservableSet; import org.jabref.logic.bibtex.FieldWriter; import org.jabref.model.database.event.EntriesAddedEvent; @@ -29,6 +30,7 @@ import org.jabref.model.entry.Month; import org.jabref.model.entry.event.EntriesEventSource; import org.jabref.model.entry.event.EntryChangedEvent; +import org.jabref.model.entry.event.FieldAddedOrRemovedEvent; import org.jabref.model.entry.event.FieldChangedEvent; import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.FieldFactory; @@ -52,6 +54,8 @@ public class BibDatabase { * State attributes */ private final ObservableList entries = FXCollections.synchronizedObservableList(FXCollections.observableArrayList(BibEntry::getObservables)); + + private final ObservableSet visibleFields = FXCollections.observableSet(); private Map bibtexStrings = new ConcurrentHashMap<>(); private final EventBus eventBus = new EventBus(); @@ -136,13 +140,8 @@ public ObservableList getEntries() { * * @return set of fieldnames, that are visible */ - public Set getAllVisibleFields() { - Set allFields = new TreeSet<>(Comparator.comparing(Field::getName)); - for (BibEntry e : getEntries()) { - allFields.addAll(e.getFields()); - } - return allFields.stream().filter(field -> !FieldFactory.isInternalField(field)) - .collect(Collectors.toSet()); + public ObservableSet getAllVisibleFields() { + return visibleFields; } /** @@ -214,6 +213,8 @@ public synchronized void insertEntries(List newEntries, EntriesEventSo eventBus.post(new EntriesAddedEvent(newEntries, newEntries.get(0), eventSource)); } entries.addAll(newEntries); + + updateVisibleFields(); } public synchronized void removeEntry(BibEntry bibEntry) { @@ -251,6 +252,7 @@ public synchronized void removeEntries(List toBeDeleted, EntriesEventS boolean anyRemoved = entries.removeIf(entry -> ids.contains(entry.getId())); if (anyRemoved) { eventBus.post(new EntriesRemovedEvent(toBeDeleted, eventSource)); + updateVisibleFields(); } } @@ -584,6 +586,30 @@ private void relayEntryChangeEvent(FieldChangedEvent event) { eventBus.post(event); } + @Subscribe + private void listen(FieldAddedOrRemovedEvent event) { + // When a field is removed from an entry we can't tell if it's + // still present in other entries, and thus we can't remove it + // from the set of visible fields. However, when a new field is added + // to any entry, we can simply add it to the set because we're + // going to add it whether other entries have it or not + boolean isAdded = visibleFields.add(event.getField()); + if (!isAdded) { + updateVisibleFields(); + } + } + + private void updateVisibleFields() { + visibleFields.clear(); + Set allFields = new TreeSet<>(Comparator.comparing(Field::getName)); + for (BibEntry e : getEntries()) { + allFields.addAll(e.getFields()); + } + visibleFields.addAll(allFields.stream().filter(field -> !FieldFactory.isInternalField(field)) + .filter(field -> StringUtil.isNotBlank(field.getName())) + .collect(Collectors.toSet())); + } + public Optional getReferencedEntry(BibEntry entry) { return entry.getField(StandardField.CROSSREF).flatMap(this::getEntryByCitationKey); } diff --git a/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java b/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java index b737596abd6..836cd39dac8 100644 --- a/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java +++ b/src/main/java/org/jabref/model/entry/BibEntryTypesManager.java @@ -11,6 +11,7 @@ import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.field.BibField; import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.types.BiblatexAPAEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexEntryTypeDefinitions; import org.jabref.model.entry.types.BiblatexSoftwareEntryTypeDefinitions; import org.jabref.model.entry.types.BibtexEntryTypeDefinitions; @@ -21,7 +22,7 @@ public class BibEntryTypesManager { public static final String ENTRYTYPE_FLAG = "jabref-entrytype: "; private final InternalEntryTypes BIBTEX = new InternalEntryTypes(Stream.concat(BibtexEntryTypeDefinitions.ALL.stream(), IEEETranEntryTypeDefinitions.ALL.stream()).collect(Collectors.toList())); - private final InternalEntryTypes BIBLATEX = new InternalEntryTypes(Stream.concat(BiblatexEntryTypeDefinitions.ALL.stream(), BiblatexSoftwareEntryTypeDefinitions.ALL.stream()).collect(Collectors.toList())); + private final InternalEntryTypes BIBLATEX = new InternalEntryTypes(Stream.concat(BiblatexEntryTypeDefinitions.ALL.stream(), Stream.concat(BiblatexSoftwareEntryTypeDefinitions.ALL.stream(), BiblatexAPAEntryTypeDefinitions.ALL.stream())).collect(Collectors.toList())); public BibEntryTypesManager() { } @@ -99,6 +100,7 @@ public List getAllCustomTypes(BibDatabaseMode mode) { return customizedTypes.stream() .filter(entryType -> BiblatexEntryTypeDefinitions.ALL.stream().noneMatch(biblatexType -> biblatexType.getType().equals(entryType.getType()))) .filter(entryType -> BiblatexSoftwareEntryTypeDefinitions.ALL.stream().noneMatch(biblatexSoftware -> biblatexSoftware.getType().equals(entryType.getType()))) + .filter(entryType -> BiblatexAPAEntryTypeDefinitions.ALL.stream().noneMatch(biblatexAPA -> biblatexAPA.getType().equals(entryType.getType()))) .collect(Collectors.toList()); } } diff --git a/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java b/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java new file mode 100644 index 00000000000..b2938963efe --- /dev/null +++ b/src/main/java/org/jabref/model/entry/field/BiblatexApaField.java @@ -0,0 +1,82 @@ +package org.jabref.model.entry.field; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; +import java.util.Optional; +import java.util.Set; + +import org.jabref.model.entry.types.BiblatexApaEntryType; + +public enum BiblatexApaField implements Field { + + AMENDMENT("amendment"), + ARTICLE("article"), + CITATION("citation"), + CITATION_CITEORG("citation_citeorg"), + CITATION_CITEDATE("citation_citedate", FieldProperty.DATE), + CITATION_CITEINFO("citation_citeinfo"), + SECTION("section", FieldProperty.NUMERIC), + SOURCE("source"); + + private final String name; + private final String displayName; + private final Set properties; + + BiblatexApaField(String name) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexApaField(String name, String displayName) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexApaField(String name, String displayName, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.of(first, rest); + } + + BiblatexApaField(String name, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.of(first, rest); + } + + public static Optional fromName(T type, String name) { + if (!(type instanceof BiblatexApaEntryType)) { + return Optional.empty(); + } + return Arrays.stream(BiblatexApaField.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public Set getProperties() { + return Collections.unmodifiableSet(properties); + } + + @Override + public String getName() { + return name; + } + + @Override + public boolean isStandardField() { + return false; + } + + @Override + public String getDisplayName() { + if (displayName == null) { + return Field.super.getDisplayName(); + } else { + return displayName; + } + } +} diff --git a/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java b/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java new file mode 100644 index 00000000000..fc3bb4dcc48 --- /dev/null +++ b/src/main/java/org/jabref/model/entry/field/BiblatexSoftwareField.java @@ -0,0 +1,82 @@ +package org.jabref.model.entry.field; + +import java.util.Arrays; +import java.util.Collections; +import java.util.EnumSet; +import java.util.Optional; +import java.util.Set; + +import org.jabref.model.entry.types.BiblatexSoftwareEntryType; + +public enum BiblatexSoftwareField implements Field { + + HALID("hal_id"), + HALVERSION("hal_version"), + INTRODUCEDIN("introducedin"), + LICENSE("license"), + RELATEDTYPE("relatedtype"), + RELATEDSTRING("relatedstring"), + REPOSITORY("repository"), + SWHID("swhid"); + + private final String name; + private final String displayName; + private final Set properties; + + BiblatexSoftwareField(String name) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexSoftwareField(String name, String displayName) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.noneOf(FieldProperty.class); + } + + BiblatexSoftwareField(String name, String displayName, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = displayName; + this.properties = EnumSet.of(first, rest); + } + + BiblatexSoftwareField(String name, FieldProperty first, FieldProperty... rest) { + this.name = name; + this.displayName = null; + this.properties = EnumSet.of(first, rest); + } + + public static Optional fromName(T type, String name) { + if (!(type instanceof BiblatexSoftwareEntryType)) { + return Optional.empty(); + } + return Arrays.stream(BiblatexSoftwareField.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public Set getProperties() { + return Collections.unmodifiableSet(properties); + } + + @Override + public String getName() { + return name; + } + + @Override + public boolean isStandardField() { + return false; + } + + @Override + public String getDisplayName() { + if (displayName == null) { + return Field.super.getDisplayName(); + } else { + return displayName; + } + } +} diff --git a/src/main/java/org/jabref/model/entry/field/FieldFactory.java b/src/main/java/org/jabref/model/entry/field/FieldFactory.java index b1c2f3b99b9..07a34b039d3 100644 --- a/src/main/java/org/jabref/model/entry/field/FieldFactory.java +++ b/src/main/java/org/jabref/model/entry/field/FieldFactory.java @@ -73,13 +73,23 @@ public static String serializeFieldsList(Collection fields) { .collect(Collectors.joining(DELIMITER)); } + public static Field parseField(T type, String fieldName) { + return OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + OptionalUtil.orElse( + InternalField.fromName(fieldName), + StandardField.fromName(fieldName)), + SpecialField.fromName(fieldName)), + IEEEField.fromName(fieldName)), + BiblatexSoftwareField.fromName(type, fieldName)), + BiblatexApaField.fromName(type, fieldName)) + .orElse(new UnknownField(fieldName)); + } + public static Field parseField(String fieldName) { - return OptionalUtil.orElse(OptionalUtil.orElse(OptionalUtil.orElse( - InternalField.fromName(fieldName), - StandardField.fromName(fieldName)), - SpecialField.fromName(fieldName)), - IEEEField.fromName(fieldName)) - .orElse(new UnknownField(fieldName)); + return parseField(null, fieldName); } public static Set getKeyFields() { @@ -138,6 +148,8 @@ private static Set getFieldsFiltered(Predicate selector) { private static Set getAllFields() { Set fields = new HashSet<>(); + fields.addAll(EnumSet.allOf(BiblatexApaField.class)); + fields.addAll(EnumSet.allOf(BiblatexSoftwareField.class)); fields.addAll(EnumSet.allOf(IEEEField.class)); fields.addAll(EnumSet.allOf(InternalField.class)); fields.addAll(EnumSet.allOf(SpecialField.class)); diff --git a/src/main/java/org/jabref/model/entry/field/IEEEField.java b/src/main/java/org/jabref/model/entry/field/IEEEField.java index d00bec02eee..f56ce79ea99 100644 --- a/src/main/java/org/jabref/model/entry/field/IEEEField.java +++ b/src/main/java/org/jabref/model/entry/field/IEEEField.java @@ -31,7 +31,7 @@ public enum IEEEField implements Field { this.properties = EnumSet.of(first, rest); } - public static Optional fromName(String name) { + public static Optional fromName(String name) { return Arrays.stream(IEEEField.values()) .filter(field -> field.getName().equalsIgnoreCase(name)) .findAny(); diff --git a/src/main/java/org/jabref/model/entry/field/StandardField.java b/src/main/java/org/jabref/model/entry/field/StandardField.java index ef3646802e0..bcf9b710b7d 100644 --- a/src/main/java/org/jabref/model/entry/field/StandardField.java +++ b/src/main/java/org/jabref/model/entry/field/StandardField.java @@ -59,14 +59,11 @@ public enum StandardField implements Field { FOREWORD("foreword", FieldProperty.PERSON_NAMES), FOLDER("folder"), GENDER("gender", FieldProperty.GENDER), - HALID("hal_id"), - HALVERSION("hal_version"), HOLDER("holder", FieldProperty.PERSON_NAMES), HOWPUBLISHED("howpublished"), IDS("ids", FieldProperty.MULTIPLE_ENTRY_LINK), INSTITUTION("institution"), INTRODUCTION("introduction", FieldProperty.PERSON_NAMES), - INTRODUCEDIN("introducedin"), ISBN("isbn", "ISBN", FieldProperty.ISBN), ISRN("isrn", "ISRN"), ISSN("issn", "ISSN"), @@ -81,7 +78,6 @@ public enum StandardField implements Field { LANGUAGE("language", FieldProperty.LANGUAGE), LABEL("label"), LIBRARY("library"), - LICENSE("license"), LOCATION("location"), MAINSUBTITLE("mainsubtitle", FieldProperty.BOOK_NAME), MAINTITLE("maintitle", FieldProperty.BOOK_NAME), @@ -106,10 +102,7 @@ public enum StandardField implements Field { PUBSTATE("pubstate", FieldProperty.PUBLICATION_STATE), PRIMARYCLASS("primaryclass"), RELATED("related", FieldProperty.MULTIPLE_ENTRY_LINK), - RELATEDTYPE("relatedtype"), - RELATEDSTRING("relatedstring"), REPORTNO("reportno"), - REPOSITORY("repository"), REVIEW("review"), REVISION("revision"), SCHOOL("school"), @@ -120,7 +113,6 @@ public enum StandardField implements Field { SORTKEY("sortkey"), SORTNAME("sortname", FieldProperty.PERSON_NAMES), SUBTITLE("subtitle"), - SWHID("swhid"), TITLE("title"), TITLEADDON("titleaddon"), TRANSLATOR("translator", FieldProperty.PERSON_NAMES), diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java b/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java new file mode 100644 index 00000000000..f5c0b89b0be --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexAPAEntryTypeDefinitions.java @@ -0,0 +1,43 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.List; + +import org.jabref.model.entry.BibEntryType; +import org.jabref.model.entry.BibEntryTypeBuilder; +import org.jabref.model.entry.field.BiblatexApaField; +import org.jabref.model.entry.field.StandardField; + +public class BiblatexAPAEntryTypeDefinitions { + + private static final BibEntryType JURISDICTION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Jurisdiction) + .withImportantFields(StandardField.ORGANIZATION, BiblatexApaField.CITATION_CITEORG, BiblatexApaField.CITATION_CITEDATE, BiblatexApaField.CITATION_CITEDATE, StandardField.ORIGDATE) + .withRequiredFields(StandardField.TITLE, BiblatexApaField.CITATION, BiblatexApaField.CITATION_CITEINFO, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType LEGISLATION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legislation) + .withImportantFields(StandardField.TITLEADDON, StandardField.ORIGDATE) + .withRequiredFields(StandardField.TITLE, StandardField.LOCATION, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType LEGADMINMATERIAL = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legadminmaterial) + .withImportantFields(StandardField.NUMBER, StandardField.SHORTTITLE, StandardField.NOTE, StandardField.KEYWORDS) + .withRequiredFields(StandardField.TITLE, BiblatexApaField.CITATION, StandardField.URL, StandardField.DATE) + .build(); + + private static final BibEntryType CONSTITUTION = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Constitution) + .withImportantFields(BiblatexApaField.ARTICLE, BiblatexApaField.AMENDMENT, StandardField.EVENTDATE, StandardField.KEYWORDS, StandardField.PART, BiblatexApaField.SECTION) + .withRequiredFields(BiblatexApaField.SOURCE, StandardField.TYPE) + .build(); + + private static final BibEntryType LEGAL = new BibEntryTypeBuilder() + .withType(BiblatexApaEntryType.Legal) + .withRequiredFields(StandardField.TITLE, StandardField.DATE, StandardField.URI, StandardField.KEYWORDS, StandardField.PART, BiblatexApaField.SECTION) + .build(); + + public static final List ALL = Arrays.asList(JURISDICTION, LEGISLATION, LEGADMINMATERIAL, CONSTITUTION, LEGAL); +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java b/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java new file mode 100644 index 00000000000..6a1d8d3abe2 --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexApaEntryType.java @@ -0,0 +1,36 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.Locale; +import java.util.Optional; + +public enum BiblatexApaEntryType implements EntryType { + + Legislation("Legislation"), + Legadminmaterial("Legadminmaterial"), + Jurisdiction("Jurisdiction"), + Constitution("Constitution"), + Legal("Legal"); + + private final String displayName; + + BiblatexApaEntryType(String displayName) { + this.displayName = displayName; + } + + @Override + public String getName() { + return displayName.toLowerCase(Locale.ENGLISH); + } + + @Override + public String getDisplayName() { + return displayName; + } + + public static Optional fromName(String name) { + return Arrays.stream(BiblatexApaEntryType.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java new file mode 100644 index 00000000000..041c48e592a --- /dev/null +++ b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryType.java @@ -0,0 +1,35 @@ +package org.jabref.model.entry.types; + +import java.util.Arrays; +import java.util.Locale; +import java.util.Optional; + +public enum BiblatexSoftwareEntryType implements EntryType { + + Dataset("Dataset"), + SoftwareVersion("SoftwareVersion"), + SoftwareModule("SoftwareModule"), + CodeFragment("CodeFragment"); + + private final String displayName; + + BiblatexSoftwareEntryType(String displayName) { + this.displayName = displayName; + } + + public static Optional fromName(String name) { + return Arrays.stream(BiblatexSoftwareEntryType.values()) + .filter(field -> field.getName().equalsIgnoreCase(name)) + .findAny(); + } + + @Override + public String getName() { + return displayName.toLowerCase(Locale.ENGLISH); + } + + @Override + public String getDisplayName() { + return displayName; + } +} diff --git a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java index 0fc48e6d904..6a2e7f0851e 100644 --- a/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java +++ b/src/main/java/org/jabref/model/entry/types/BiblatexSoftwareEntryTypeDefinitions.java @@ -5,6 +5,7 @@ import org.jabref.model.entry.BibEntryType; import org.jabref.model.entry.BibEntryTypeBuilder; +import org.jabref.model.entry.field.BiblatexSoftwareField; import org.jabref.model.entry.field.OrFields; import org.jabref.model.entry.field.StandardField; @@ -12,40 +13,40 @@ public class BiblatexSoftwareEntryTypeDefinitions { private static final BibEntryType SOFTWARE = new BibEntryTypeBuilder() .withType(StandardEntryType.Software) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.VERSION, StandardField.YEAR) .build(); private static final BibEntryType SOFTWAREVERSION = new BibEntryTypeBuilder() - .withType(StandardEntryType.SoftwareVersion) - .withImportantFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, StandardField.HALID, StandardField.HALVERSION, - StandardField.INSTITUTION, StandardField.INTRODUCEDIN, StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, - StandardField.PUBLISHER, StandardField.RELATED, StandardField.RELATEDTYPE, StandardField.RELATEDSTRING, - StandardField.REPOSITORY, StandardField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) + .withType(BiblatexSoftwareEntryType.SoftwareVersion) + .withImportantFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, + StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, + StandardField.PUBLISHER, StandardField.RELATED, BiblatexSoftwareField.RELATEDTYPE, BiblatexSoftwareField.RELATEDSTRING, + BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.YEAR, StandardField.VERSION) - .withDetailFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, StandardField.HALID, StandardField.HALVERSION, - StandardField.INSTITUTION, StandardField.INTRODUCEDIN, StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, - StandardField.PUBLISHER, StandardField.RELATED, StandardField.RELATEDTYPE, StandardField.RELATEDSTRING, - StandardField.REPOSITORY, StandardField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) + .withDetailFields(StandardField.DATE, StandardField.EPRINTCLASS, StandardField.EPRINTTYPE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, + StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, + StandardField.PUBLISHER, StandardField.RELATED, BiblatexSoftwareField.RELATEDTYPE, BiblatexSoftwareField.RELATEDSTRING, + BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.SUBTITLE, StandardField.URLDATE) .withRequiredFields(new OrFields(StandardField.AUTHOR, StandardField.EDITOR), StandardField.TITLE, StandardField.URL, StandardField.YEAR) .build(); private static final BibEntryType SOFTWAREMODULE = new BibEntryTypeBuilder() - .withType(StandardEntryType.SoftwareModule) + .withType(BiblatexSoftwareEntryType.SoftwareModule) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(StandardField.AUTHOR, StandardField.SUBTITLE, StandardField.URL, StandardField.YEAR) .build(); private static final BibEntryType CODEFRAGMENT = new BibEntryTypeBuilder() - .withType(StandardEntryType.CodeFragment) + .withType(BiblatexSoftwareEntryType.CodeFragment) .withImportantFields(StandardField.DATE, StandardField.DOI, StandardField.EPRINTTYPE, StandardField.EPRINTCLASS, StandardField.EPRINT, - StandardField.EDITOR, StandardField.FILE, StandardField.HALID, StandardField.HALVERSION, StandardField.INSTITUTION, StandardField.INTRODUCEDIN, - StandardField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, - StandardField.RELATEDSTRING, StandardField.REPOSITORY, StandardField.SWHID, StandardField.URLDATE, StandardField.VERSION) + StandardField.EDITOR, StandardField.FILE, BiblatexSoftwareField.HALID, BiblatexSoftwareField.HALVERSION, StandardField.INSTITUTION, BiblatexSoftwareField.INTRODUCEDIN, + BiblatexSoftwareField.LICENSE, StandardField.MONTH, StandardField.NOTE, StandardField.ORGANIZATION, StandardField.PUBLISHER, StandardField.RELATED, + BiblatexSoftwareField.RELATEDSTRING, BiblatexSoftwareField.REPOSITORY, BiblatexSoftwareField.SWHID, StandardField.URLDATE, StandardField.VERSION) .withRequiredFields(StandardField.URL) .build(); diff --git a/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java b/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java index 1c94f6dd9a0..6b40a56c4f4 100644 --- a/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java +++ b/src/main/java/org/jabref/model/entry/types/EntryTypeFactory.java @@ -49,6 +49,8 @@ private static boolean isBiblatex(EntryType type) { public static EntryType parse(String typeName) { List types = new ArrayList<>(Arrays.asList(StandardEntryType.values())); types.addAll(Arrays.asList(IEEETranEntryType.values())); + types.addAll(Arrays.asList(BiblatexSoftwareEntryType.values())); + types.addAll(Arrays.asList(BiblatexApaEntryType.values())); types.addAll(Arrays.asList(SystematicLiteratureReviewStudyEntryType.values())); return types.stream().filter(type -> type.getName().equals(typeName.toLowerCase(Locale.ENGLISH))).findFirst().orElse(new UnknownEntryType(typeName)); diff --git a/src/main/java/org/jabref/model/entry/types/StandardEntryType.java b/src/main/java/org/jabref/model/entry/types/StandardEntryType.java index fb2922e61a0..6f6f91ac6a5 100644 --- a/src/main/java/org/jabref/model/entry/types/StandardEntryType.java +++ b/src/main/java/org/jabref/model/entry/types/StandardEntryType.java @@ -36,10 +36,7 @@ public enum StandardEntryType implements EntryType { Thesis("Thesis"), WWW("WWW"), Software("Software"), - Dataset("Dataset"), - SoftwareVersion("SoftwareVersion"), - SoftwareModule("SoftwareModule"), - CodeFragment("CodeFragment"); + Dataset("Dataset"); private final String displayName; diff --git a/src/main/java/org/jabref/model/metadata/MetaData.java b/src/main/java/org/jabref/model/metadata/MetaData.java index 238658803d3..9b0ed2ae856 100644 --- a/src/main/java/org/jabref/model/metadata/MetaData.java +++ b/src/main/java/org/jabref/model/metadata/MetaData.java @@ -68,6 +68,7 @@ public class MetaData { private final ContentSelectors contentSelectors = new ContentSelectors(); private final Map> unknownMetaData = new HashMap<>(); private boolean isEventPropagationEnabled = true; + private boolean encodingExplicitlySupplied; /** * Constructs an empty metadata. @@ -291,6 +292,17 @@ public void setEncoding(Charset encoding, ChangePropagation postChanges) { } } + public boolean getEncodingExplicitlySupplied() { + return encodingExplicitlySupplied; + } + + /** + * Sets the indication whether the encoding was set using "% Encoding: ..." or whether it was detected "magically" + */ + public void setEncodingExplicitlySupplied(boolean encodingExplicitlySupplied) { + this.encodingExplicitlySupplied = encodingExplicitlySupplied; + } + /** * If disabled {@link MetaDataChangedEvent} will not be posted. */ @@ -349,6 +361,7 @@ public boolean equals(Object o) { return (isProtected == metaData.isProtected) && Objects.equals(groupsRoot.getValue(), metaData.groupsRoot.getValue()) && Objects.equals(encoding, metaData.encoding) + && Objects.equals(encodingExplicitlySupplied, metaData.encodingExplicitlySupplied) && Objects.equals(saveOrderConfig, metaData.saveOrderConfig) && Objects.equals(citeKeyPatterns, metaData.citeKeyPatterns) && Objects.equals(userFileDirectory, metaData.userFileDirectory) @@ -362,7 +375,7 @@ public boolean equals(Object o) { @Override public int hashCode() { - return Objects.hash(groupsRoot.getValue(), encoding, saveOrderConfig, citeKeyPatterns, userFileDirectory, + return Objects.hash(groupsRoot.getValue(), encoding, encodingExplicitlySupplied, saveOrderConfig, citeKeyPatterns, userFileDirectory, defaultCiteKeyPattern, saveActions, mode, isProtected, defaultFileDirectory); } } diff --git a/src/main/java/org/jabref/model/strings/StringUtil.java b/src/main/java/org/jabref/model/strings/StringUtil.java index 19f0b7a4542..2a443306e4d 100644 --- a/src/main/java/org/jabref/model/strings/StringUtil.java +++ b/src/main/java/org/jabref/model/strings/StringUtil.java @@ -750,4 +750,15 @@ public static String quoteStringIfSpaceIsContained(String string) { return string; } } + + /** + * Checks if the given string contains any whitespace characters. The supported whitespace characters + * are the set of characters matched by {@code \s} in regular expressions, which are {@code [ \t\n\x0B\f\r]}. + * + * @param s The string to check + * @return {@code True} if the given string does contain at least one whitespace character, {@code False} otherwise + * */ + public static boolean containsWhitespace(String s) { + return s.chars().anyMatch(Character::isWhitespace); + } } diff --git a/src/main/resources/l10n/JabRef_da.properties b/src/main/resources/l10n/JabRef_da.properties index c07cb809a45..97eeb562207 100644 --- a/src/main/resources/l10n/JabRef_da.properties +++ b/src/main/resources/l10n/JabRef_da.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 matcher regulærudtrykket %1 %0\ matches\ the\ term\ %1=%0 matcher udtrykket %1 @@ -122,8 +123,6 @@ Cite\ command=Citations-kommando Clear=Ryd -Clear\ fields=Ryd felter - Close\ dialog=Luk dialog @@ -574,11 +573,18 @@ Remove\ entry\ from\ import=Fjern post fra import Remove\ group=Fjern gruppe -Remove\ group\ "%0"?=Fjern gruppen "%0"? +Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Fjern gruppen "%0" og dens undergrupper? -Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernede gruppen "%0" og dens undergrupper + + Remove\ link=Slet link @@ -587,8 +593,6 @@ Remove\ old\ entry=Fjern gammel post Removed\ group\ "%0".=Fjernede gruppen "%0" -Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernede gruppen "%0" og dens undergrupper - Removed\ string=Streng fjernet Renamed\ string=ændrede navn på streng @@ -642,9 +646,6 @@ Select\ file\ from\ ZIP-archive=Vælg fil fra ZIP-fil Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Vælg forgreningerne for at inspicere og acceptere eller forkaste ændringer -Set\ field=Sæt felt -Set\ fields=Sæt felter - Settings=Indstillinger Shortcut=Genvej @@ -675,7 +676,6 @@ Size=Størrelse Skipped\ -\ No\ PDF\ linked=Sprang over - ingen PDF-fil linket Skipped\ -\ PDF\ does\ not\ exist=Sprang over - PDF-filen findes ikke -Skipped\ entry.=Sprang over post. source\ edit=redigering af kilde @@ -787,8 +787,6 @@ MIME\ type=MIME-type This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ running\ instance\ of\ JabRef\ instead\ of\ opening\ a\ new\ instance.\ For\ instance,\ this\ is\ useful\ when\ you\ open\ a\ file\ in\ JabRef\ from\ your\ web\ browser.\ Note\ that\ this\ will\ prevent\ you\ from\ running\ more\ than\ one\ instance\ of\ JabRef\ at\ a\ time.=Denne funktion tillader, at flere filer kan åbnes eller importeres i en allerede kørende JabRef i stedet for at åbne programmet påny. For eksempel er dette praktisk, når du åbner filer i JabRef fra din web browser. Bemærk at dette vil forhindre dig i at køre mere end en instans af JabRef ad gangen. Rename\ field=Omdøb felt -Rename\ field\ to=Omdøb felt til -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flyt indhold af et felt til et felt med et andet navn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan ikke bruge port %0 til fjernstyring; et andet program bruger den måske. Prøv en anden port. @@ -914,6 +912,7 @@ Finished\ automatically\ setting\ external\ links.=Fuldførte automatisk udfyldn + Open\ folder=Åben mappe @@ -1088,6 +1087,8 @@ Default\ pattern=Standardmønster + + diff --git a/src/main/resources/l10n/JabRef_de.properties b/src/main/resources/l10n/JabRef_de.properties index d9f4e1272f6..ee962e6a73b 100644 --- a/src/main/resources/l10n/JabRef_de.properties +++ b/src/main/resources/l10n/JabRef_de.properties @@ -19,7 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 Einträge -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 Export erfolgreich. Ordner mit der gespeicherten Datei öffnen? +Export\ operation\ finished\ successfully.=Exportvorgang erfolgreich abgeschlossen. + +Reveal\ in\ File\ Explorer=Im Datei-Explorer anzeigen %0\ matches\ the\ regular\ expression\ %1=%0 exakt dem regulären Ausdruck %1 entspricht @@ -151,8 +153,6 @@ Cite\ command=Cite-Befehl Clear=Zurücksetzen -Clear\ fields=Felder löschen - Open\ /\ close\ entry\ editor=Eintragseditor öffnen / schließen Close\ dialog=Dialog schließen @@ -703,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Ausgewählte Einträge aus dieser G Remove\ group=Gruppe löschen -Remove\ group\ "%0"?=Gruppe "%0" löschen? +Remove\ group\ and\ subgroups=Gruppe und Untergruppen löschen + +Remove\ groups\ and\ subgroups=Gruppen und Untergruppen entfernen + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Alle ausgewählten Gruppen entfernen und ihre Untergruppen beibehalten? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Gruppe "%0" entfernen und Untergruppen beibehalten? + +Remove\ groups=Gruppen entfernen + +Removed\ all\ selected\ groups.=Alle ausgewählten Gruppen entfernen. Remove\ group\ "%0"\ and\ its\ subgroups?=Gruppe "%0" inklusive Untergruppen löschen? -Remove\ group\ and\ subgroups=Gruppe und Untergruppen löschen +Removed\ group\ "%0"\ and\ its\ subgroups.=Gruppe "%0" inklusive Untergruppen gelöscht. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Alle ausgewählten Gruppen und ihre Untergruppen entfernen? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Alle ausgewählten Gruppen und ihre Untergruppen entfernt. Remove\ link=Link löschen @@ -717,8 +731,6 @@ Remove\ string\ %0=Entferne String %0 Removed\ group\ "%0".=Gruppe "%0" gelöscht. -Removed\ group\ "%0"\ and\ its\ subgroups.=Gruppe "%0" inklusive Untergruppen gelöscht. - Removed\ string=String gelöscht Renamed\ string=String umbenannt @@ -790,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Eintrag aus der ZIP-Archiv auswählen Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Wählen Sie die Verzweigungen aus, um die Änderungen zu sehen und anzunehmen oder zu verwerfen -Set\ field=Setze Feld -Set\ fields=Felder setzen - Settings=Einstellungen Shortcut=Tastenkürzel @@ -825,7 +834,12 @@ Size=Größe Skipped\ -\ No\ PDF\ linked=Übersprungen - Kein PDF verlinkt Skipped\ -\ PDF\ does\ not\ exist=Übersprungen - PDF exisitert nicht -Skipped\ entry.=Eintrag übersprungen. +JabRef\ skipped\ the\ entry.=JabRef hat den Eintrag übersprungen. +Import\ error=Fehler beim Importieren +Open\ library\ error=Fehler beim Öffnen der Bibliothek +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Bitte überprüfen Sie Ihre Bibliotheksdatei auf falsche Syntax. +SourceTab\ error=SourceTab Fehler +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Benutzereingabe via Eintragseditor im Tab `{}bibtex source` führte zu einem Fehler. Sort\ subgroups=Untergruppen sortieren @@ -972,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Zeile %0\: Be No\ full\ text\ document\ found=Kein Volltext-Dokument gefunden Download\ from\ URL=Download von URL Rename\ field=Feld umbenennen -Append\ field=An Feld anfügen -Append\ to\ fields=An Felder anfügen -Rename\ field\ to=Feld umbenennen -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Inhalt eines Felds in ein Feld mit anderem Namen verschieben Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Port %0 konnte nicht für externen Zugriff genutzt werden; er wird möglicherweise von einer anderen Anwendung benutzt. Versuchen Sie einen anderen Port. @@ -1243,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Benutzerdefinierte Proxy-Konfiguration verwend Proxy\ requires\ authentication=Proxy erfordert Authentifizierung Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Achtung\: Das Passwort wird im Klartext gespeichert\! Clear\ connection\ settings=Verbindungseinstellungen zurücksetzen +Check\ Proxy\ Setting=Überprüfe Proxy-Einstellungen +Check\ connection=Verbindung prüfen +Connection\ failed\!=Verbindung fehlgeschlagen\! +Connection\ successful\!=Verbindung erfolgreich\! + +SSL\ Configuration=SSL-Konfiguration +SSL\ configuration\ changed=SSL-Konfiguration geändert +SSL\ certificate\ file=SSL-Zertifikatsdatei +Duplicate\ Certificates=Doppelte Zertifikate +You\ already\ added\ this\ certificate=Sie haben dieses Zertifikat bereits hinzugefügt Open\ folder=Ordner öffnen Export\ sort\ order=Sortierreihenfolge exportieren @@ -1279,6 +1299,7 @@ Please\ open\ %0\ manually.=Bitte öffnen Sie %0 manuell. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Der Link wurde in die Zwischenablage kopiert. Open\ %0\ file=%0 Datei öffnen +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Konsole konnte nicht automatisch erkannt werden. Bitte definieren Sie eine eigene Konsolenapplikation in den Einstellungen. Cannot\ delete\ file=Datei kann nicht gelöscht werden. File\ permission\ error=Fehler beim Dateizugriff. @@ -1666,6 +1687,8 @@ Issue\ report\ successful=Problemreport war erfolgreich Your\ issue\ was\ reported\ in\ your\ browser.=Ihr Problem wurde im Browser gemeldet. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Die Log und Ausnahmefehler-Informationen wurden in die Zwischenablage kopiert. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Bitte fügen Sie die Informationen (mit STRG+V) in die Problembeschreibung ein. +Last\ notification=Letzte Benachrichtigung +Check\ the\ event\ log\ to\ see\ all\ notifications=Überprüfe das Ereignisprotokoll, um alle Benachrichtigungen zu sehen Host=Host Port=Port @@ -1703,7 +1726,6 @@ Protected\ terms\ file=Geschützte Begriffsdatei Style\ file=Stildatei Open\ OpenOffice/LibreOffice\ connection=Öffne OpenOffice/LibreOffice Verbindung -You\ must\ enter\ at\ least\ one\ field\ name=Sie müssen mindestens einen Feldnamen angeben Non-ASCII\ encoded\ character\ found=Nicht ASCII-kodiertes Zeichen gefunden Non-UTF-8\ encoded\ field\ found=Nicht UTF-8 kodiertes Feld gefunden Toggle\ web\ search\ interface=Websuche ein-/ausschalten @@ -1841,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Entfernen aller Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Konnte Daten von '%0' nicht abrufen. Entry\ from\ %0\ could\ not\ be\ parsed.=Eintrag von %0 konnte nicht analysiert werden. Invalid\ identifier\:\ '%0'.=Ungültige Kennung\: "%0". -This\ paper\ has\ been\ withdrawn.=Dieses Paper wurde zurückgezogen. empty\ citation\ key=leerer Zitationsschlüssel Aux\ file=Aux-Datei Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Gruppe mit Einträgen die in einer vorhanden Tex-Datei zitiert worden sind @@ -2324,10 +2345,6 @@ Removes\ digits.=Ziffern entfernen. Presets=Voreinstellungen -Check\ Proxy\ Setting=Überprüfe Proxy-Einstellungen -Check\ connection=Verbindung prüfen -Connection\ failed\!=Verbindung fehlgeschlagen\! -Connection\ successful\!=Verbindung erfolgreich\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Gruppen aus Schlüsselwörtern im folgenden Feld generieren Generate\ groups\ for\ author\ last\ names=Gruppen für Nachnamen der Autoren generieren Regular\ expression=Regulärere Ausdruck @@ -2400,7 +2417,6 @@ Entry\ Type=Eintragstyp Entry\ types=Eintragstypen Field\ names=Feldnamen Others=Andere -Overwrite\ existing\ field\ values=Bestehende Feldwerte überschreiben Recommended=Empfohlen Authors\ and\ Title=Autoren und Titel @@ -2445,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Remote-Dienste Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Erlaube das Senden von PDF-Dateien und Rohzitaten an einen JabRef Online-Dienst (Grobid) um Metadaten zu ermitteln. Dies führt zu besseren Ergebnissen. +Fetcher\ cannot\ be\ tested\!=Der Fetcher kann nicht getestet werden\! +Fetcher\ unknown\!=Fetcher unbekannt\! Character\ by\ character=Zeichen für Zeichen Embedded=Eingebettet @@ -2468,26 +2486,32 @@ Valid\ from=Gültig seit Valid\ to=Gültig bis Signature\ algorithm=Signaturalgorithmus Version=Version -SSL\ Configuration=SSL-Konfiguration -SSL\ certificate\ file=SSL-Zertifikatsdatei -Duplicate\ Certificates=Doppelte Zertifikate -You\ already\ added\ this\ certificate=Sie haben dieses Zertifikat bereits hinzugefügt Error\ downloading=Fehler beim Herunterladen Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Fehler beim Schreiben von Metadaten. Details finden Sie im Fehlerprotokoll. Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metadaten können nicht geschrieben werden, Datei %1 nicht gefunden. Success\!\ Finished\ writing\ metadata.=Erfolgreich\! Das Schreiben der Metadaten ist abgeschlossen. -Overwrite\ Non\ empty\ fields=Overwrite Non empty fields -Set=Set -Append=Append -Clear\ field=Clear field -field\ value=field value -Edit\ field\ value\ of\ selected\ entries=Edit field value of selected entries -Rename=Rename -new\ field\ name=new field name -Copy\ value=Copy value -Move\ value=Move value -Swap\ values=Swap values -Copy\ or\ move\ the\ value\ of\ one\ field\ to\ another=Copy or move the value of one field to another +Custom\ API\ key=Eigener API-Schlüssel +Check\ %0\ API\ Key\ Setting=%0 API-Schlüsseleinstellungen überprüfen + +Edit\ content=Inhalt ändern +Copy\ or\ Move\ content=Inhalt kopieren oder verschieben +Overwrite\ field\ content=Inhalt des Feldes überschreiben +Set=Festlegen +Append=Anfügen +Clear\ field\ content=Feldinhalt löschen +Set\ or\ append\ content=Setze oder füge Inhalt an +Edit\ field\ content\ for\ selected\ entries=Feldinhalt der ausgewählten Einträge bearbeiten +Rename=Umbenennen +New\ field\ name=Neuer Feldname +Copy\ content=Inhalt kopieren +Move\ content=Inhalt verschieben +Swap\ content=Inhalte vertauschen +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Inhalt eines Feldes kopieren oder in ein anderes Feld verschieben +Automatic\ field\ editor=Automatischer Feldeditor + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Hinweis\: Wenn es den ursprünglichen Einträgen an Schlüsselwörtern fehlt, die sich für die neue Gruppenkonfiguration qualifizieren, wird die Bestätigung hier diese hinzufügen) +Assign=Zuweisen +Do\ not\ assign=Nicht zuweisen diff --git a/src/main/resources/l10n/JabRef_el.properties b/src/main/resources/l10n/JabRef_el.properties index 290a41f97b6..e3f04f56ffe 100644 --- a/src/main/resources/l10n/JabRef_el.properties +++ b/src/main/resources/l10n/JabRef_el.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 ταιριάζει την συνήθη έκφραση %1 %0\ matches\ the\ term\ %1=%0 ταιριάζει τον όρο %1 @@ -126,8 +127,6 @@ Cite\ command=Εντολή αναφοράς Clear=Καθαρισμός -Clear\ fields=Καθαρισμός πεδίων - Close\ dialog=Κλείσμο διαλόγου @@ -608,11 +607,18 @@ Remove\ selected\ entries\ from\ this\ group=Αφαίρεση επιλεγμέν Remove\ group=Αφαίρεση ομάδας -Remove\ group\ "%0"?=Αφαίρεση της ομάδας "%0"; +Remove\ group\ and\ subgroups=Αφαίρεση ομάδας και υπο-ομάδων + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Αφαίρεση της ομάδας "%0" και των υπο-ομάδων της; -Remove\ group\ and\ subgroups=Αφαίρεση ομάδας και υπο-ομάδων +Removed\ group\ "%0"\ and\ its\ subgroups.=Η ομάδα "%0" και οι υπο-ομάδες της έχουν αφαιρεθεί. + + Remove\ link=Αφαίρεση συνδέσμου @@ -621,8 +627,6 @@ Remove\ old\ entry=Αφαίρεση παλαιάς καταχώρησης Removed\ group\ "%0".=Η ομάδα "%0" έχει αφαιρεθεί. -Removed\ group\ "%0"\ and\ its\ subgroups.=Η ομάδα "%0" και οι υπο-ομάδες της έχουν αφαιρεθεί. - Removed\ string=Η συμβολοσειρά έχει αφαιρεθεί Renamed\ string=Η συμβολοσειρά έχει μετονομαστεί @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Επιλογή αρχείου από συμπι Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Επιλέξτε τους δενδροειδείς κόμβους για να προβάλλετε και να αποδεχτείτε ή να απορρίψετε αλλαγές -Set\ field=Ορισμός πεδίου -Set\ fields=Ορισμός πεδίων - Settings=Ρυθμίσεις Shortcut=Συντόμευση @@ -714,7 +715,6 @@ Size=Μέγεθος Skipped\ -\ No\ PDF\ linked=Παραβλέφθηκε - Δεν έχει συνδεθεί PDF Skipped\ -\ PDF\ does\ not\ exist=Παραβλέφθηκε - Δεν υπάρχει PDF -Skipped\ entry.=Η καταχώρηση παραβλέφθηκε. source\ edit=επεξεργασία πηγής @@ -829,10 +829,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru No\ full\ text\ document\ found=Δεν βρέθηκε αρχείο πλήρους κειμένου Download\ from\ URL=Κατέβασμα από σύνδεσμο URL Rename\ field=Μετονομασία πεδίου -Append\ field=Προσάρτηση πεδίου -Append\ to\ fields=Προσάρτηση στα πεδία -Rename\ field\ to=Μετονομασία πεδίου ως -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Μετακίνηση περιεχομένων ενός πεδίου σε ένα πεδίο με διαφορετικό όνομα Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Αδύνατη η χρήση της θύρας %0 για απομακρυσμένη λειτουργία, ίσως χρησιμοποιείται από μια άλλη εφαρμογή. Δοκιμάστε να ορίσετε άλλη θύρα. @@ -1046,6 +1042,7 @@ Proxy\ requires\ authentication=Ο διακομιστής μεσολάβησης Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Προσοχή\: Ο κωδικός πρόσβασης είναι αποθηκευμένος σε απλό κείμενο\! Clear\ connection\ settings=Εκκαθάριση ρυθμίσεων σύνδεσης + Open\ folder=Άνοιγμα φακέλου Export\ sort\ order=Εξαγωγή σειράς ταξινόμησης @@ -1448,7 +1445,6 @@ Protected\ terms\ file=Αρχείο προστατευμένων όρων Style\ file=Αρχείο στυλ Open\ OpenOffice/LibreOffice\ connection=Άνοιγμα σύνδεσης OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Πρέπει να καταχωρήσετε τουλάχιστον ένα όνομα πεδίου Non-ASCII\ encoded\ character\ found=Βρέθηκε χαρακτήρας χωρίς κωδικοποίηση ASCII Toggle\ web\ search\ interface=Εναλλαγή διεπαφής αναζήτησης στο διαδίκτυο @@ -1569,7 +1565,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Αφαιρεί Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Αδυναμία ανάκτησης δεδομένων καταχώρησης από '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Αδυναμία ανάλυσης καταχώρησης από %0. Invalid\ identifier\:\ '%0'.=Μη έγκυρο αναγνωριστικό\: '%0'. -This\ paper\ has\ been\ withdrawn.=Αυτή η εργασία έχει αποσυρθεί. Aux\ file=Αρχείο AUX Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Ομάδα που περιέχει καταχωρήσεις που αναφέρονται σε ένα συγκεκριμένο αρχείο TeX @@ -1702,6 +1697,8 @@ Default\ pattern=Προεπιλεγμένο μοτίβο + + diff --git a/src/main/resources/l10n/JabRef_en.properties b/src/main/resources/l10n/JabRef_en.properties index 5ae7dae2bab..ab98eefe526 100644 --- a/src/main/resources/l10n/JabRef_en.properties +++ b/src/main/resources/l10n/JabRef_en.properties @@ -833,7 +833,12 @@ Size=Size Skipped\ -\ No\ PDF\ linked=Skipped - No PDF linked Skipped\ -\ PDF\ does\ not\ exist=Skipped - PDF does not exist -Skipped\ entry.=Skipped entry. +JabRef\ skipped\ the\ entry.=JabRef skipped the entry. +Import\ error=Import error +Open\ library\ error=Open library error +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Please check your library file for wrong syntax. +SourceTab\ error=SourceTab error +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=User input via entry-editor in `{}bibtex source` tab led to failure. Sort\ subgroups=Sort subgroups @@ -2484,20 +2489,20 @@ Success\!\ Finished\ writing\ metadata.=Success! Finished writing metadata. Custom\ API\ key=Custom API key Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting -Edit\ field\ value=Edit field value -Two\ fields=Two fields -Overwrite\ Non\ empty\ fields=Overwrite Non empty fields +Edit\ content=Edit content +Copy\ or\ Move\ content=Copy or Move content +Overwrite\ field\ content=Overwrite field content Set=Set Append=Append -Clear\ field=Clear field -Field\ value=Field value -Edit\ field\ value\ of\ selected\ entries=Edit field value of selected entries +Clear\ field\ content=Clear field content +Set\ or\ append\ content=Set or append content +Edit\ field\ content\ for\ selected\ entries=Edit field content for selected entries Rename=Rename New\ field\ name=New field name -Copy\ value=Copy value -Move\ value=Move value -Swap\ values=Swap values -Copy\ or\ move\ the\ value\ of\ one\ field\ to\ another=Copy or move the value of one field to another +Copy\ content=Copy content +Move\ content=Move content +Swap\ content=Swap content +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copy or move the content of one field to another Automatic\ field\ editor=Automatic field editor Open\ Link=Open Link diff --git a/src/main/resources/l10n/JabRef_es.properties b/src/main/resources/l10n/JabRef_es.properties index c70107490a3..60e35d98ad0 100644 --- a/src/main/resources/l10n/JabRef_es.properties +++ b/src/main/resources/l10n/JabRef_es.properties @@ -1,7 +1,12 @@ +Could\ not\ delete\ empty\ entries.=No se pudieron eliminar entradas vacías. +Delete\ empty\ entries=Eliminar entradas vacías +Empty\ entries=Vaciar entradas +Keep\ empty\ entries=Mantener entradas vacías +Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=La biblioteca '%0' tiene entradas vacías. ¿Quieres eliminarlas? Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ restart.\ You\ may\ encounter\ errors\ if\ you\ continue\ with\ this\ session.=No es posible supervisar los cambios en los archivos. Cierre los archivos y procesos y reinicie. Puede que se produzcan errores si continúa con esta sesión. %0\ contains\ the\ regular\ expression\ %1=%0 contiene la expresión regular %1 @@ -15,6 +20,8 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entradas +Reveal\ in\ File\ Explorer=Mostrar en Explorador de Archivos + %0\ matches\ the\ regular\ expression\ %1=%0 coincidencias con la Expresión Regular %1 %0\ matches\ the\ term\ %1=%0 coincidencias con el término %1 @@ -48,6 +55,7 @@ The\ path\ need\ not\ be\ on\ the\ classpath\ of\ JabRef.=La ruta no debe estar Add\ a\ regular\ expression\ for\ the\ key\ pattern.=Añadir una expresión regular para el patrón clave. +Add\ entry\ manually=Añadir entrada manualmente Add\ selected\ entries\ to\ this\ group=Añadir entradas seleccionadas a este grupo @@ -144,8 +152,6 @@ Cite\ command=Comando Citar Clear=Limpiar -Clear\ fields=Limpiar campos - Open\ /\ close\ entry\ editor=Abrir o cerrar editor de entradas Close\ dialog=Cerrar diálogo @@ -167,6 +173,8 @@ Copy=Copiar Copy\ title=Copiar título Copy\ \\cite{citation\ key}=Copiar \\cite{citation key} +Copy\ citation\ (html)=Copiar cita (html) +Copy\ citation\ (text)=Copiar cita (texto) Copy\ citation\ key=Copiar clave de cita Copy\ citation\ key\ and\ link=Copiar clave de cita y enlace Copy\ citation\ key\ and\ title=Copiar clave y título de cita @@ -215,13 +223,17 @@ cut\ entries=Cortar entradas cut\ entry\ %0=corte de entrada %0 +DOI\ not\ found=DOI no encontrado Library\ encoding=Codificación de la biblioteca Library\ properties=Propiedades de la biblioteca +%0\ -\ Library\ properties=%0 - Propiedades de la biblioteca Default=Por defecto +Character\ encoding\ UTF-8\ is\ not\ supported.=La codificación de caracteres UTF-8 no restá soportada. +UTF-8\ could\ not\ be\ used\ to\ encode\ the\ following\ characters\:\ %0=UTF-8 no se pudo utilizar para codificar los siguientes caracteres\: %0 The\ chosen\ encoding\ '%0'\ could\ not\ encode\ the\ following\ characters\:=La codificación de caracteres '%' no puede codificar los siguientes caracteres\: Downloading=Descargando @@ -270,6 +282,7 @@ Downloaded\ website\ as\ an\ HTML\ file.=Se descargó el sitio web como un archi duplicate\ removal=eliminación de duplicados +Duplicate\ fields=Campos duplicados Duplicate\ string\ name=Nombre de cadena duplicado @@ -331,6 +344,7 @@ External\ file\ links=Enlaces a archivos externos External\ programs=Programas externos +Failed\ to\ import\ by\ ID=Error al importar por ID Field=Campo @@ -355,6 +369,7 @@ Filter=Filtro Filter\ groups=Filtros +Finished\ writing\ metadata\ for\ %0\ file\ (%1\ skipped,\ %2\ errors).=Escritura de metadatos para el archivo %0 finalizada (%1 omitidos, %2 errores). First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=En primer lugar, seleccione las entradas para las que desea generar claves @@ -371,6 +386,7 @@ Formatter\ name=Nombre del formateador found\ in\ AUX\ file=encontrado en archivo AUX +Fulltext\ search=Búsqueda de texto completo Fulltext\ for=Texto completo de @@ -449,6 +465,9 @@ Include\ subgroups\:\ When\ selected,\ view\ entries\ contained\ in\ this\ group Independent\ group\:\ When\ selected,\ view\ only\ this\ group's\ entries=Grupo independiente\: ver sólo las entradas de este grupo cuando esté seleccionado. I\ Agree=Acepto +Indexing\ pdf\ files=Indexando archivos pdf +Indexing\ for\ %0=Indexando para %0 +%0\ of\ %1\ linked\ files\ added\ to\ the\ index=%0 de %1 archivos enlazados añadidos al índice Invalid\ citation\ key=La clave de cita no es válida @@ -462,6 +481,8 @@ JabRef\ requests\ recommendations\ from\ Mr.\ DLib,\ which\ is\ an\ external\ se JabRef\ Version\ (Required\ to\ ensure\ backwards\ compatibility\ with\ Mr.\ DLib's\ Web\ Service)=Versión JabRef (necesaria para asegurar la compatibilidad con el servicio web de Mr. DLib) Journal\ abbreviations=Abreviaturas de publicaciones +Journal\ lists\:=Listados de revistas\: +Remove\ journal\ '%0'=Eliminar revista '%0' Keep\ both=Mantener ambos @@ -497,6 +518,7 @@ Main\ file\ directory=Carpeta del archivo principal Manage\ custom\ exports=Administrar exportaciones personalizadas Manage\ custom\ imports=Administrar importaciones personalizadas +External\ file\ types=Tipos de archivos externos Mark\ new\ entries\ with\ owner\ name=Marcar nuevas entradas con nombre de propietario @@ -519,7 +541,9 @@ Moved\ group\ "%0".=Se ha movido el grupo "%0". Mr.\ DLib\ Privacy\ settings=Mr. DLib Configuración de la privacidad +No\ database\ is\ open=No hay ninguna base de datos abierta +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=Necesitamos una base de datos desde la que exportar. Abrir una. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=No se han recibido recomendaciones del Mr. DLib para esta entrada. @@ -553,6 +577,7 @@ No\ journal\ names\ could\ be\ abbreviated.=No se pudieron abreviar nombres de r No\ journal\ names\ could\ be\ unabbreviated.=No se pudieron expandir nombres de revistas. +No\ DOI\ data\ exists=No existen datos DOI not=no @@ -677,11 +702,25 @@ Remove\ selected\ entries\ from\ this\ group=Eliminar las entradas seleccionadas Remove\ group=Eliminar grupo -Remove\ group\ "%0"?=¿Eliminar el grupo "%0%? +Remove\ group\ and\ subgroups=Eliminar grupo y subgrupos + +Remove\ groups\ and\ subgroups=Eliminar grupos y subgrupos + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=¿Eliminar todos los grupos seleccionados y mantener sus subgrupos? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=¿Eliminar grupo "%0" y mantener sus subgrupos? + +Remove\ groups=Eliminar grupos + +Removed\ all\ selected\ groups.=Eliminados todos los grupos seleccionados. Remove\ group\ "%0"\ and\ its\ subgroups?=¿Eliminar el grupo "%0" y sus subgrupos? -Remove\ group\ and\ subgroups=Eliminar grupo y subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Se ha eliminado el grupo "%0" y sus subgrupos. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=¿Eliminar todos los grupos seleccionados y sus subgrupos? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Eliminados todos los grupos seleccionados y sus subgrupos. Remove\ link=Eliminar enlace @@ -691,8 +730,6 @@ Remove\ string\ %0=Eliminar cadena %0 Removed\ group\ "%0".=el grupo "%0" se ha eliminado. -Removed\ group\ "%0"\ and\ its\ subgroups.=Se ha eliminado el grupo "%0" y sus subgrupos. - Removed\ string=Cadena eliminada Renamed\ string=Cadena renombrada @@ -715,6 +752,8 @@ Replaces\ Unicode\ ligatures\ with\ their\ expanded\ form=Reemplaza las ligadura Required\ fields=Campos requeridos +Do\ not\ resolve\ BibTeX\ strings=No resolver cadenas BibTeX +Resolve\ BibTeX\ strings\ for\ the\ following\ fields=Resolver las cadenas BibTeX para los siguientes campos resolved=resuelto @@ -722,11 +761,13 @@ Restart=Reiniciar Restart\ required=Reinicio requerido +Return\ to\ dialog=Volver al diálogo Review=Revisar Review\ changes=Revisar cambios Review\ Field\ Migration=Revisar campo de migración +Loading=Cargando Save=Guardar Save\ all\ finished.=Guardar todos los finalizados @@ -740,6 +781,7 @@ Save\ library\ as...=Guardar biblioteca como... Saving=Guardando Saving\ all\ libraries...=Guardando todas las bibliotecas... Saving\ library=Guardando biblioteca +Library\ saved=Biblioteca guardada Saved\ selected\ to\ '%0'.=Selección guardada en '%0'. Search=Buscar @@ -759,9 +801,6 @@ Select\ file\ from\ ZIP-archive=Seleccionar archivo desde archivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Seleccionar nodos de árbol para ver y aceptar o rechazar los cambios. -Set\ field=Establecer campo -Set\ fields=Establecer campos - Settings=Ajustes Shortcut=Atajo @@ -794,7 +833,12 @@ Size=Tamaño Skipped\ -\ No\ PDF\ linked=Omitido - No se enlazó PDF Skipped\ -\ PDF\ does\ not\ exist=Omitido - No existe el PDF -Skipped\ entry.=Entrada omitida. +JabRef\ skipped\ the\ entry.=JabRef omitió la entrada. +Import\ error=Error al importar +Open\ library\ error=Error al abrir la biblioteca +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Por favor, comprueba errores de sintaxis en tu archivo de biblioteca. +SourceTab\ error=Error de SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Entrada de usuario a través del editor de entrada en la pestaña `{}bibtex source` condujo al error. Sort\ subgroups=Ordenar subgrupos @@ -883,13 +927,18 @@ Warning=Advertencia Warnings=Advertencias +Warning\:\ You\ added\ field\ "%0"\ twice.\ Only\ one\ will\ be\ kept.=Advertencia\: Añadiste el campo "%0" dos veces. Solo uno será utilizado. web\ link=enlace a web What\ do\ you\ want\ to\ do?=¿Qué desea hacer? Whatever\ option\ you\ choose,\ Mr.\ DLib\ may\ share\ its\ data\ with\ research\ partners\ to\ further\ improve\ recommendation\ quality\ as\ part\ of\ a\ 'living\ lab'.\ Mr.\ DLib\ may\ also\ release\ public\ datasets\ that\ may\ contain\ anonymized\ information\ about\ you\ and\ the\ recommendations\ (sensitive\ information\ such\ as\ metadata\ of\ your\ articles\ will\ be\ anonymised\ through\ e.g.\ hashing).\ Research\ partners\ are\ obliged\ to\ adhere\ to\ the\ same\ strict\ data\ protection\ policy\ as\ Mr.\ DLib.=Sea cual sea la opción que elija, Mr. DLib puede compartir sus datos con socios de investigación para mejorar aún más la calidad de las recomendaciones como parte de un "laboratorio vivo". Mr. DLib también puede publicar conjuntos de datos públicos que pueden contener información anónima sobre usted y las recomendaciones (la información confidencial como los metadatos de sus artículos será anonimizada a través de, por ejemplo, hashing). Los socios de investigación están obligados a adherirse a la misma estricta política de protección de datos que Mr. DLib. +Will\ write\ metadata\ to\ the\ PDFs\ linked\ from\ selected\ entries.=Escribirá metadatos en los PDFs enlazados desde entradas seleccionadas. +Write\ BibTeXEntry\ as\ metadata\ to\ PDF.=Escribir entrada BibTeX como metadatos XMP en el PDF. +Write\ metadata\ for\ all\ PDFs\ in\ current\ library?=¿Escribir metadatos para todos los PDFs de la biblioteca actual? +Writing\ metadata\ for\ selected\ entries...=Escribiendo metadatos para las entradas seleccionadas... Write\ BibTeXEntry\ as\ XMP\ metadata\ to\ PDF.=Escribe BibTeXEntry como metadatos XMP en los PDF. @@ -929,10 +978,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Renglón %0\: No\ full\ text\ document\ found=No se encontró un documento de texto completo Download\ from\ URL=Descargar desde URL Rename\ field=Renombrar campo -Append\ field=Añadir campo -Append\ to\ fields=Añadir a campos -Rename\ field\ to=Renombrar campo a -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover contenidos de un campo en un campo con nombre diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=No se puede usar el puerto %0 para operación remota\: Puede estar en uso por otra aplicación. Pruebe a especificar otro puerto. @@ -1182,6 +1227,11 @@ Use\ custom\ proxy\ configuration=Usar configuración de proxy personalizada Proxy\ requires\ authentication=El proxi requiere autenticación Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atención\: ¡La contraseña está almacenada como texto plano\! Clear\ connection\ settings=Limpiar ajustes de conexión +Check\ Proxy\ Setting=Comprobar configuración de «proxy» +Check\ connection=Comprobar conexión +Connection\ failed\!=Falló la conexión. +Connection\ successful\!=Conexión correcta. + Open\ folder=Abrir pasta Export\ sort\ order=Criterio de ordenación para exportación @@ -1586,6 +1636,8 @@ Issue\ report\ successful=Comunicación de problema satisfactoria Your\ issue\ was\ reported\ in\ your\ browser.=Su problema fue comunicado a través del navegador The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=La información de excepción y registro fue copiada a su portapapeles Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Por favor, pegue esta información (con Ctrl+V) en la descripción del problema. +Last\ notification=Última notificación +Check\ the\ event\ log\ to\ see\ all\ notifications=Revisa el registro de eventos para ver todas las notificaciones Host=Host/Servidor Port=Puerto @@ -1623,7 +1675,6 @@ Protected\ terms\ file=Archivo de términos protegidos Style\ file=Archivo de estilo Open\ OpenOffice/LibreOffice\ connection=Conexión OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Debe introducir, al menos, un nombre de campo Non-ASCII\ encoded\ character\ found=Se han encontrado caracteres con codificación no-ASCII Non-UTF-8\ encoded\ field\ found=Se encontró un campo no codificado en UTF-8 Toggle\ web\ search\ interface=Cambiar interfaz de búsqueda web @@ -1732,6 +1783,7 @@ Delete\ '%0'=Eliminar '%0' Delete\ from\ disk=Eliminar del disco duro Remove\ from\ entry=Eliminar de la entrada There\ exists\ already\ a\ group\ with\ the\ same\ name.=Ya existe un grupo con el mismo nombre. +If\ you\ use\ it,\ it\ will\ inherit\ all\ entries\ from\ this\ other\ group.=Si lo utiliza, heredará todas las entradas de este otro grupo. Copy\ linked\ file=Copiar archivo enlazado Copy\ linked\ file\ to\ folder...=Copiar archivo enlazado a la carpeta... @@ -1760,7 +1812,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Elimina todos l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=No se pueden recuperar los datos de la entrada desde '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=No se pudo analizar la entrada de %0. Invalid\ identifier\:\ '%0'.=Identificador no válido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Este artículo ha sido retirado. empty\ citation\ key=clave de cita vacía Aux\ file=Archivo Aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo que contiene las entradas citadas en un archivo TeX determinado @@ -1788,6 +1839,7 @@ Could\ not\ connect\ to\ Vim\ server.\ Make\ sure\ that\ Vim\ is\ running\ with\ Could\ not\ connect\ to\ a\ running\ gnuserv\ process.\ Make\ sure\ that\ Emacs\ or\ XEmacs\ is\ running,\ and\ that\ the\ server\ has\ been\ started\ (by\ running\ the\ command\ 'server-start'/'gnuserv-start').=No se puede conectar con un proceso gnuserv en ejecución. Asegúrese de que Emacs o XEmacs se está ejecutando y de que el servidor ha sido iniciado (ejecutando el comando 'server-start'/'gnuserv-start'). Error\ pushing\ entries=Error al enviar entradas +Preamble=Preámbulo Markings=Marcados Use\ selected\ instance=Usar la instancia seleccionada @@ -1806,6 +1858,7 @@ Blog=Blog Check\ integrity=Verificar integridad Cleanup\ URL\ link=Limpiar un enlace URL Cleanup\ URL\ link\ by\ removing\ special\ symbols\ and\ extracting\ simple\ link=Limpiar un enlace URL eliminando los símbolos especiales y extrayendo un enlace simple +Copy\ DOI=Copiar DOI Copy\ DOI\ url=Copiar la url del DOI Development\ version=Versión de desarrollo Export\ selected\ entries=Exportar registros seleccionados @@ -1815,6 +1868,8 @@ JabRef\ resources=Recursos sobre JabRef Manage\ journal\ abbreviations=Administrar abreviaturas de publicaciones Manage\ protected\ terms=gestionar términos protegidos New\ entry\ from\ plain\ text=Nueva entrada desde texto sin formato +Import\ by\ ID=Importar por ID +Enter\ a\ valid\ ID=Introduzca un ID válido New\ sublibrary\ based\ on\ AUX\ file=Nueva subbiblioteca a partir de un archivo AUX Push\ entries\ to\ external\ application\ (%0)=Agregar registros a aplicación externa (%0) Quit=Salir @@ -1874,6 +1929,7 @@ Keyword\ separator=Separador de palabras clave Remove\ keyword=Eliminar palabra clave Are\ you\ sure\ you\ want\ to\ remove\ keyword\:\ "%0"?=¿Está seguro de que desea eliminar la palabra clave\: "%0"? Reset\ to\ default=Restablecer los valores por defecto +String\ constants=Constantes de cadena Export\ all\ entries=Exportar todas las entradas Generate\ citation\ keys=Generar claves de cita Manage\ field\ names\ &\ content=Gestionar nombres y contenido de los campos @@ -2003,6 +2059,7 @@ Please\ provide\ a\ valid\ aux\ file.=Por favor, proporcione un archivo AUX vál Keyword\ delimiter=Separador de palabras clave Hierarchical\ keyword\ delimiter=Separador de palabras clave jerárquicas Escape\ ampersands=Escape de ampersands +Escape\ dollar\ sign=Símbolo de escape dólar Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Consejo\:\n\nPara buscar Pedro en todos los campos, escriba\:\npedro\n\nPara buscar Pedro en el campo author y eléctrico en el campo title, escriba\:\nauthor\=Pedro and title\=eléctrico @@ -2212,7 +2269,11 @@ Reveal\ in\ file\ explorer=Revelar en el explorador de archivos Autolink\ files=Enlazar archivos automáticamente +Custom\ editor\ tabs=Pestañas de editor personalizadas +Custom\ export\ formats=Formatos de exportación personalizados +Custom\ import\ formats=Formatos de importación personalizados +No\ list\ enabled=No hay lista habilitada Protect\ selection=Proteger selección Customized\ preview\ style=Estilo de previsualización personalizado @@ -2233,17 +2294,23 @@ Removes\ digits.=Elimina dígitos. Presets=Preconfiguraciones -Check\ Proxy\ Setting=Comprobar configuración de «proxy» -Check\ connection=Comprobar conexión -Connection\ failed\!=Falló la conexión. -Connection\ successful\!=Conexión correcta. Generate\ groups\ from\ keywords\ in\ the\ following\ field=Generar grupos a partir de palabras clave en el campo siguiente Generate\ groups\ for\ author\ last\ names=Generar grupos para apellidos de autores Regular\ expression=Expresión regular Error\ importing.\ See\ the\ error\ log\ for\ details.=Error al importar. Consulte el registro de errores para obtener detalles. +Error\ from\ import\:\ %0=Error de importación\: %0 +Error\ reading\ PDF\ content\:\ %0=Error al leer el contenido PDF\: %0 +Importing\ bib\ entry=Importando entrada bibliográfica +Importing\ using\ extracted\ PDF\ data=Importación usando datos extraídos del PDF +No\ BibTeX\ data\ found.\ Creating\ empty\ entry\ with\ file\ link=No se encontraron datos BibTeX. Creando entrada vacía con enlace de archivo +No\ metadata\ found.\ Creating\ empty\ entry\ with\ file\ link=No se encontraron metadatos. Creando entrada vacía con enlace de archivo +Processing\ file\ %0=Procesando archivo %0 +Export\ selected=Exportar seleccionados +Separate\ merged\ citations=Citas fusionadas separadas +Separate\ citations=Citas separadas Custom\ DOI\ URI=URI personalizado de DOI @@ -2272,7 +2339,6 @@ Entry\ Type=Tipo de entrada Entry\ types=Tipos de entrada Field\ names=Nombre del campo Others=Otros -Overwrite\ existing\ field\ values=Sobreescribir valores de campo existentes Recommended=Recomendado Authors\ and\ Title=Autoría y título @@ -2303,3 +2369,5 @@ Question=Pregunta + + diff --git a/src/main/resources/l10n/JabRef_fa.properties b/src/main/resources/l10n/JabRef_fa.properties index 9fabe5b9655..fb60c5076e1 100644 --- a/src/main/resources/l10n/JabRef_fa.properties +++ b/src/main/resources/l10n/JabRef_fa.properties @@ -14,6 +14,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ + Abbreviate\ names=نام های مخفف Abbreviated\ %0\ journal\ names.=نام های مخفف شده %0 ژورنال. @@ -98,8 +99,6 @@ Case\ sensitive=حساس به کوچکی و بزرگی حروف Clear=پاکسازی -Clear\ fields=پاک کردن فیلدها - @@ -375,13 +374,18 @@ Open\ terminal\ here=در اینجا پایانه را باز کن -Library\ saved=کتابخانه ذخیره شد +Library\ saved=کتابخانه ذخیره شد + + + + + @@ -496,6 +500,7 @@ Merge\ entries=ترکیب کردن ورودیها + Open\ folder=بازکردن پوشه @@ -665,3 +670,5 @@ Search\ results\ from\ open\ libraries=جستجوی نتایج از کتابخا + + diff --git a/src/main/resources/l10n/JabRef_fr.properties b/src/main/resources/l10n/JabRef_fr.properties index 425843db382..46481f71dce 100644 --- a/src/main/resources/l10n/JabRef_fr.properties +++ b/src/main/resources/l10n/JabRef_fr.properties @@ -19,7 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entrées -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=Exportation de %0 réussie. Voulez-vous ouvrir le dossier contenant le fichier sauvegardé ? +Export\ operation\ finished\ successfully.=L'opération d'export s'est terminée avec succès. + +Reveal\ in\ File\ Explorer=Montrer dans l'explorateur de fichiers %0\ matches\ the\ regular\ expression\ %1=%0 correspond à l'expression régulière %1 @@ -151,8 +153,6 @@ Cite\ command=Commande Cite Clear=Vider -Clear\ fields=Vider les champs - Open\ /\ close\ entry\ editor=Ouvrir/fermer l'éditeur d'entrées Close\ dialog=Fermer la fenêtre @@ -295,7 +295,7 @@ Dynamically\ group\ entries\ by\ searching\ a\ field\ for\ a\ keyword=Grouper dy Each\ line\ must\ be\ of\ the\ following\ form\:\ 'tab\:field1;field2;...;fieldN'.=Chaque ligne doit être au format suivant \: 'tab\:field1;field2;...;fieldN'. -Edit=Éditer +Edit=Édition Edit\ file\ type=Modifier le type de fichier @@ -703,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Supprimer de ce groupe les entrées Remove\ group=Supprimer le groupe -Remove\ group\ "%0"?=Supprimer le groupe « %0 » ? +Remove\ group\ and\ subgroups=Supprimer le groupe et les sous-groupes + +Remove\ groups\ and\ subgroups=Supprimer les groupes et les sous-groupes + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Supprimer tous les groupes sélectionnés et conserver leurs sous-groupes ? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Supprimer le groupe « %0 » et conserver ses sous-groupes ? + +Remove\ groups=Supprimer les groupes + +Removed\ all\ selected\ groups.=Tous les groupes sélectionnés ont été supprimés. Remove\ group\ "%0"\ and\ its\ subgroups?=Supprimer le groupe « %0 » et ses sous-groupes ? -Remove\ group\ and\ subgroups=Supprimer le groupe et les sous-groupes +Removed\ group\ "%0"\ and\ its\ subgroups.=Groupe « %0 » et ses sous-groupes supprimés. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Supprimer tous les groupes sélectionnés et leurs sous-groupes ? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Tous les groupes sélectionnés et leurs sous-groupes ont été supprimés. Remove\ link=Supprimer le lien @@ -717,8 +731,6 @@ Remove\ string\ %0=Supprimer la chaîne %0 Removed\ group\ "%0".=Groupe « %0 » supprimé. -Removed\ group\ "%0"\ and\ its\ subgroups.=Groupe « %0 » et ses sous-groupes supprimés. - Removed\ string=Chaîne supprimée Renamed\ string=Chaîne renommée @@ -770,7 +782,7 @@ Save\ library\ as...=Enregistrer le fichier sous... Saving=Enregistrement en cours Saving\ all\ libraries...=Enregistrement de tous les fichiers... Saving\ library=Enregistrement du fichier en cours -Library\ saved=Fichier sauvé +Library\ saved=Fichier enregistré Saved\ selected\ to\ '%0'.=Sélection enregistrée dans « %0 ». Search=Recherche @@ -790,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Sélectionner un fichier depuis une archive ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Sélectionner les nœuds de l'arborescence pour voir, et accepter ou rejeter, les modifications -Set\ field=Configurer le champ -Set\ fields=Configurer les champs - Settings=Paramètres Shortcut=Raccourci @@ -825,7 +834,12 @@ Size=Taille Skipped\ -\ No\ PDF\ linked=Sauté - Pas de PDF lié Skipped\ -\ PDF\ does\ not\ exist=Omis - Le PDF n'existe pas -Skipped\ entry.=Entrée omise. +JabRef\ skipped\ the\ entry.=JabRef a ignoré l'entrée. +Import\ error=Erreur d'importation +Open\ library\ error=Erreur d'ouverture du fichier +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Veuillez vérifier la syntaxe de votre fichier bibliographique. +SourceTab\ error=Erreur de SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=La saisie de l'utilisateur via l'éditeur d'entrée dans l'onglet `{}bibtex source` a conduit à un échec. Sort\ subgroups=Trier les sous-groupes @@ -971,18 +985,14 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Ligne Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Ligne %0 \: clef de citation corrompue %1 (virgule manquante). No\ full\ text\ document\ found=Aucun texte intégral trouvé Download\ from\ URL=Télécharger depuis l'URL -Rename\ field=Renommer le champ -Append\ field=Ajouter au champ -Append\ to\ fields=Ajouter aux champs -Rename\ field\ to=Renommer le champ en -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Déplacer le contenu d'un champ vers un champ d'un nom différent +Rename\ field=Renommer Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Le port %0 ne peut pas être utilisé pour une opération à distance ; un autre logiciel pourrait être en train de l'utiliser. Essayer de spécifier un autre port. Looking\ for\ full\ text\ document...=Recherche du texte intégral... A\ local\ copy\ will\ be\ opened.=Une copie locale sera ouverte. -Autosave\ local\ libraries=Sauvegarder automatiquement les fichiers locaux -Automatically\ save\ the\ library\ to=Sauvegarder automatiquement les fichiers dans \: +Autosave\ local\ libraries=Enregistrer automatiquement les fichiers locaux +Automatically\ save\ the\ library\ to=Enregistrer automatiquement les fichiers dans \: Please\ enter\ a\ valid\ file\ path.=Veuillez saisir un chemin de fichier valide. Error\ opening\ file\ '%0'.=Erreur lors de l'ouverture du fichier « %0 ». @@ -1003,7 +1013,7 @@ contains=contient search\ expression=expression de recherche Optional\ fields\ 2=Champs optionnels 2 -Waiting\ for\ save\ operation\ to\ finish=Attente de la fin de l'opération de sauvegarde +Waiting\ for\ save\ operation\ to\ finish=Attente de la fin de l'opération d'enregistrement Waiting\ for\ background\ tasks\ to\ finish.\ Quit\ anyway?=En attente de la fin des tâches en arrière-plan. Quitter quand même ? Find\ and\ remove\ duplicate\ citation\ keys=Rechercher et supprimer les clefs de citation dupliquées @@ -1037,7 +1047,7 @@ Output\ file\ missing=Fichier de sortie manquant No\ search\ matches.=Recherche sans correspondance. The\ output\ option\ depends\ on\ a\ valid\ input\ option.=L'option de sortie dépend d'une option d'entrée valide. Linked\ file\ name\ conventions=Conventions pour les noms de fichiers liés -Filename\ format\ pattern=Modèle de format de nom de fichier +Filename\ format\ pattern=Modèle de format de nom de fichier Additional\ parameters=Paramètres additionnels Cite\ selected\ entries\ between\ parenthesis=Citer les entrées sélectionnées entre parenthèses Cite\ selected\ entries\ with\ in-text\ citation=Citer les entrées sélectionnées comme incluse dans le texte @@ -1243,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Utiliser une configuration de proxy personnali Proxy\ requires\ authentication=Le proxy nécessite une authentification Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Attention \: Le mot de passe est stocké en clair dans du texte brut \! Clear\ connection\ settings=Réinitialiser les paramètres de connexion +Check\ Proxy\ Setting=Vérifier la configuration du proxy +Check\ connection=Vérifier la connexion +Connection\ failed\!=Échec de la connexion \! +Connection\ successful\!=Connexion réussie \! + +SSL\ Configuration=Configuration SSL +SSL\ configuration\ changed=Configuration SSL modifiée +SSL\ certificate\ file=Fichier de certificat SSL +Duplicate\ Certificates=Dupliquer les certificats +You\ already\ added\ this\ certificate=Vous avez déjà ajouté ce certificat Open\ folder=Ouvrir le répertoire Export\ sort\ order=Exporter l'ordre de tri @@ -1279,6 +1299,7 @@ Please\ open\ %0\ manually.=Veuillez ouvrir manuellement %0 . The\ link\ has\ been\ copied\ to\ the\ clipboard.=Le lien a été copié dans le presse-papiers. Open\ %0\ file=Ouvrir le fichier %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Impossible de détecter le terminal automatiquement. Veuillez définir un terminal personnalisé dans les préférences. Cannot\ delete\ file=Le fichier ne peut pas être supprimé File\ permission\ error=Erreur due aux permissions du fichier @@ -1666,6 +1687,8 @@ Issue\ report\ successful=Signalement de l'anomalie réussi Your\ issue\ was\ reported\ in\ your\ browser.=Votre anomalie a été affichée dans votre navigateur. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Le journal et les informations d'anomalie ont été copiées dans votre presse-papier. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Veuillez coller ces informations (avec Ctrl+V) dans la description de l'anomalie. +Last\ notification=Dernière notification +Check\ the\ event\ log\ to\ see\ all\ notifications=Consultez le journal des événements pour voir toutes les notifications Host=Hôte Port=Port @@ -1703,7 +1726,6 @@ Protected\ terms\ file=Fichier des termes protégés Style\ file=Fichier de style Open\ OpenOffice/LibreOffice\ connection=Ouvrir une connexion OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Vous devez saisir au moins un nom de champ Non-ASCII\ encoded\ character\ found=Caractère ayant un encodage non-ASCII trouvé Non-UTF-8\ encoded\ field\ found=Un champ non encodé en UTF-8 a été trouvé Toggle\ web\ search\ interface=Afficher/Masquer l'interface de recherche internet @@ -1841,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Supprime du con Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Impossible de récupérer les données sur l'entrée depuis « %0 ». Entry\ from\ %0\ could\ not\ be\ parsed.=Entrée de %0 n’a pas pu être analysée. Invalid\ identifier\:\ '%0'.=Identifiant non valide \: « %0 ». -This\ paper\ has\ been\ withdrawn.=Cet article a été retiré. empty\ citation\ key=clef de citation vide Aux\ file=Fichier AUX Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Groupe contenant les entrées citées dans un fichier TeX spécifique @@ -2144,7 +2165,7 @@ Search\ ShortScience=Recherche ShortScience Unable\ to\ open\ ShortScience.=Impossible de se connecter à ShortScience. Shared\ database=Base de données partagée -Lookup=Rechercher +Lookup=Recherche Please\ enter\ a\ field\ name\ to\ search\ for\ a\ keyword.=Veuillez entrer le nom du champ dans lequel chercher un mot-clef. Access\ date\ of\ the\ address\ specified\ in\ the\ url\ field.=Date d'accès de l'adresse spécifiée dans le champ "url". @@ -2324,10 +2345,6 @@ Removes\ digits.=Supprime les chiffres. Presets=Préréglages -Check\ Proxy\ Setting=Vérifier la configuration du proxy -Check\ connection=Vérifier la connexion -Connection\ failed\!=Échec de la connexion \! -Connection\ successful\!=Connexion réussie \! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Créer des groupes à partir des mot-clefs du champ suivant Generate\ groups\ for\ author\ last\ names=Créer des groupes à partir des noms propres des auteurs Regular\ expression=Expression régulière @@ -2400,7 +2417,6 @@ Entry\ Type=Type d’entrée Entry\ types=Types d'entrée Field\ names=Noms de champs Others=Autres -Overwrite\ existing\ field\ values=Écraser les valeurs des champs existants Recommended=Usuels Authors\ and\ Title=Auteurs et titre @@ -2470,10 +2486,6 @@ Valid\ from=Valable à partir de Valid\ to=Valide jusqu'au Signature\ algorithm=Algorithme de signature Version=Version -SSL\ Configuration=Configuration SSL -SSL\ certificate\ file=Fichier de certificat SSL -Duplicate\ Certificates=Dupliquer les certificats -You\ already\ added\ this\ certificate=Vous avez déjà ajouté ce certificat Error\ downloading=Erreur de téléchargement @@ -2483,3 +2495,23 @@ Success\!\ Finished\ writing\ metadata.=Succès \! Écriture des métadonnées t Custom\ API\ key=Clef d'API personnalisée Check\ %0\ API\ Key\ Setting=Vérifier les paramètres de la clef d'API %0 + +Edit\ content=Modifier le contenu +Copy\ or\ Move\ content=Copier ou déplacer le contenu +Overwrite\ field\ content=Écraser le contenu du champ +Set=Définir +Append=Ajouter +Clear\ field\ content=Effacer le contenu du champ +Set\ or\ append\ content=Définir ou ajouter du contenu +Edit\ field\ content\ for\ selected\ entries=Modifier le contenu d'un champ +Rename=Renommer +New\ field\ name=Nouveau nom de champ +Copy\ content=Copier le contenu +Move\ content=Déplacer le contenu +Swap\ content=Permuter le contenu +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copier ou déplacer le contenu d'un champ vers un autre +Automatic\ field\ editor=Éditeur automatique de champs + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note \: si les entrées originales n'ont pas de mots-clefs correspondant à la nouvelle configuration du groupe, confirmer ici les ajoutera) +Assign=Assigner +Do\ not\ assign=Ne pas assigner diff --git a/src/main/resources/l10n/JabRef_id.properties b/src/main/resources/l10n/JabRef_id.properties index 0e582b7cb21..6458e06d8e0 100644 --- a/src/main/resources/l10n/JabRef_id.properties +++ b/src/main/resources/l10n/JabRef_id.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 sesuai dengan Ekspresi Reguler %1 %0\ matches\ the\ term\ %1=%0 sesuai dengan istilah %1 @@ -125,8 +126,6 @@ Cite\ command=Perintah acuan Clear=Bersihkan -Clear\ fields=Bersihkan beberapa bidang - Close\ dialog=Tutup dialog @@ -604,11 +603,18 @@ Remove\ selected\ entries\ from\ this\ group=Menghapus entri yang dipilih dari g Remove\ group=Hapus grup -Remove\ group\ "%0"?=Hapus grup "%0"? +Remove\ group\ and\ subgroups=Hapus grup dan sub-grup + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Hapus grup "%0" dan sub-grup nya? -Remove\ group\ and\ subgroups=Hapus grup dan sub-grup +Removed\ group\ "%0"\ and\ its\ subgroups.=Hapus grup "%0" dan sub-grup nya. + + Remove\ link=Hapus tautan @@ -617,8 +623,6 @@ Remove\ old\ entry=Hapus entri lama Removed\ group\ "%0".=Hapus grup "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Hapus grup "%0" dan sub-grup nya. - Removed\ string=Hapus string Renamed\ string=Ganti nama string @@ -672,9 +676,6 @@ Select\ file\ from\ ZIP-archive=Pilih berkas dari arsip ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Pilih tiga nodal untuk melihat, menerima atau menolak perubahan -Set\ field=Pilih bidang -Set\ fields=Pilih beberapa bidang - Settings=Pengaturan Shortcut=Pintasan @@ -705,7 +706,6 @@ Size=Ukuran Skipped\ -\ No\ PDF\ linked=Dilompati - Tanpa tautan PDF Skipped\ -\ PDF\ does\ not\ exist=Dilompati - PDF tidak ada -Skipped\ entry.=Entri dilompati. source\ edit=sunting sumber @@ -819,10 +819,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Unduh dari URL Rename\ field=Ganti nama bidang -Append\ field=Tambahkan bidang -Append\ to\ fields=Tambahkan ke bidang -Rename\ field\ to=Ganti nama bidang menjadi -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Pindah isi dari bidang ke bidang lain dengan nama lain Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Tidak bisa memakai port %0 untuk operasi jauh; aplikasi lain mungkin sedang menggunakan. Coba port lain. @@ -1036,6 +1032,7 @@ Proxy\ requires\ authentication=Proxy memerlukan otentikasi Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Perhatian\: Kata sandi disimpan sebagai teks biasa\! Clear\ connection\ settings=Bersihkan pengaturan koneksi + Open\ folder=Buka direktori Export\ sort\ order=Ekspor urutan penyortiran @@ -1430,7 +1427,6 @@ Protected\ terms\ file=Istilah yang dilindungi Style\ file=File gaya Open\ OpenOffice/LibreOffice\ connection=Buka koneksi OpenOffice / LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Anda harus memasukkan setidaknya satu nama field Non-ASCII\ encoded\ character\ found=Karakter yang dikodekan non-ASCII ditemukan Toggle\ web\ search\ interface=Alihkan antarmuka pencarian web @@ -1551,7 +1547,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Menghapus semua Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Tidak dapat mengambil data entri dari ' % 0 '. Entry\ from\ %0\ could\ not\ be\ parsed.=Entri dari % 0 tidak dapat diuraikan. Invalid\ identifier\:\ '%0'.=Pengenal tidak valid\: ' % 0 '. -This\ paper\ has\ been\ withdrawn.=Makalah ini telah ditarik. @@ -1649,6 +1644,8 @@ Default\ pattern=Pola bawaan + + diff --git a/src/main/resources/l10n/JabRef_it.properties b/src/main/resources/l10n/JabRef_it.properties index 6f249bb1436..fc937b51fa7 100644 --- a/src/main/resources/l10n/JabRef_it.properties +++ b/src/main/resources/l10n/JabRef_it.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 voci +Export\ operation\ finished\ successfully.=L'operazione di esportazione si è conclusa con successo. + +Reveal\ in\ File\ Explorer=Mostra nel File Explorer %0\ matches\ the\ regular\ expression\ %1=%0 corrisponde all'espressione regolare %1 @@ -150,8 +153,6 @@ Cite\ command=Comando Cite Clear=Svuota -Clear\ fields=Annulla i campi - Open\ /\ close\ entry\ editor=Apri / chiudi editor voci Close\ dialog=Chiudi la finestra di dialogo @@ -700,11 +701,25 @@ Remove\ selected\ entries\ from\ this\ group=Rimuovi da questo gruppo le voci se Remove\ group=Rimuovi gruppo -Remove\ group\ "%0"?=Rimuovere il gruppo "%0"? +Remove\ group\ and\ subgroups=Rimuovi gruppo e sottogruppi + +Remove\ groups\ and\ subgroups=Rimuovi gruppi e sottogruppi + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Rimuovere tutti i gruppi selezionati e mantenere i loro sottogruppi? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Rimuovere il gruppo "%0" e mantenere i suoi sottogruppi? + +Remove\ groups=Rimuovere i gruppi + +Removed\ all\ selected\ groups.=Rimossi tutti i gruppi selezionati. Remove\ group\ "%0"\ and\ its\ subgroups?=Rimuovere il gruppo "%0" ed i suoi sottogruppi? -Remove\ group\ and\ subgroups=Rimuovi gruppo e sottogruppi +Removed\ group\ "%0"\ and\ its\ subgroups.=Rimosso gruppo "%0" e suoi sottogruppi. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Rimuovere tutti i gruppi selezionati e i loro sottogruppi? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Rimossi tutti i gruppi selezionati e i loro sottogruppi. Remove\ link=Rimuovere il collegamento @@ -714,8 +729,6 @@ Remove\ string\ %0=Rimuovi la stringa %0 Removed\ group\ "%0".=Rimosso gruppo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Rimosso gruppo "%0" e suoi sottogruppi. - Removed\ string=Stringa rimossa Renamed\ string=Stringa rinominata @@ -787,9 +800,6 @@ Select\ file\ from\ ZIP-archive=Seleziona un file da un archivio ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selezionare i nodi dell'albero per vedere ed accettare o rifiutare le modifiche -Set\ field=Imposta il campo -Set\ fields=Imposta i campi - Settings=Parametri Shortcut=Scorciatoia @@ -822,7 +832,12 @@ Size=Dimensione Skipped\ -\ No\ PDF\ linked=Saltato - Nessun file PDF collegato Skipped\ -\ PDF\ does\ not\ exist=Saltato - Il file PDF non esiste -Skipped\ entry.=Voce saltata +JabRef\ skipped\ the\ entry.=JabRef ha saltato la voce. +Import\ error=Errore di importazione +Open\ library\ error=Errore di apertura libreria +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Controlla il file della tua libreria per sintassi sbagliata. +SourceTab\ error=Errore SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=L'input dell'utente tramite entry-editor nella scheda `{}bibtex source` ha portato al fallimento. Sort\ subgroups=Ordina sottogruppi @@ -969,10 +984,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Riga %0\: Tro No\ full\ text\ document\ found=Nessun documento di testo completo trovato Download\ from\ URL=Scarica dall'URL Rename\ field=Rinomina il campo -Append\ field=Aggiungi campo -Append\ to\ fields=Aggiungi ai campi -Rename\ field\ to=Rinomina il campo in -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Sposta il contenuto di un campo in un campo con nome diverso Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Impossibile utilizzare la porta %0 per operazioni remote; la porta potrebbe essere in uso da parte di un'altra applicazione. Provare a specificare una porta diversa. @@ -1228,6 +1239,12 @@ Use\ custom\ proxy\ configuration=Utilizza una configurazione del proxy personal Proxy\ requires\ authentication=Il proxy richiede un'autenticazione Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Attenzione\: Password salvata come testo semplice\! Clear\ connection\ settings=Reinizializza i parametri di connessione +Check\ Proxy\ Setting=Controlla Impostazioni Proxy +Check\ connection=Controlla la connessione +Connection\ failed\!=Connessione fallita\! +Connection\ successful\!=Connessione riuscita\! + +SSL\ configuration\ changed=Configurazione SSL modificata Open\ folder=Apri cartella Export\ sort\ order=Esporta il modo di ordinamento @@ -1264,6 +1281,7 @@ Please\ open\ %0\ manually.=Per favore aprire %0 manualmente. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Il link è stato copiato negli appunti. Open\ %0\ file=Apri il file %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Impossibile rilevare il terminale automaticamente. Si prega di definire un terminale personalizzato nelle preferenze. Cannot\ delete\ file=Non posso cancellare il file File\ permission\ error=Errore nei permessi del file @@ -1640,6 +1658,8 @@ Issue\ report\ successful=Segnalazione del problema avvenuta con successo Your\ issue\ was\ reported\ in\ your\ browser.=Il problema è stato segnalato nel browser. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Il log e le informazioni dell'eccezione sono stati copiati negli appunti. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Per favore, incolla questa informazione (con Ctrl+V) nella descrizione del problema. +Last\ notification=Ultima notifica +Check\ the\ event\ log\ to\ see\ all\ notifications=Controlla il registro eventi per vedere tutte le notifiche Host=Host Port=Porta @@ -1677,7 +1697,6 @@ Protected\ terms\ file=File di termini protetti Style\ file=File stile Open\ OpenOffice/LibreOffice\ connection=Apri connessione OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Devi inserire almeno il nome di un campo Non-ASCII\ encoded\ character\ found=Trovato un carattere non codificato ASCII Non-UTF-8\ encoded\ field\ found=Trovato campo codificato non-UTF-8 Toggle\ web\ search\ interface=Inverti la selezione dell'interfaccia di ricerca web @@ -1815,7 +1834,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Rimuove tutte l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Impossibile recuperare dati in ingresso da '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Non è stato possibile analizzare la voce %0. Invalid\ identifier\:\ '%0'.=Identificatore non valido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Questo documento è stato ritirato. empty\ citation\ key=chiave BibTeX vuota Aux\ file=File aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Gruppo contenente voci citate in un determinato file TeX @@ -2298,10 +2316,6 @@ Removes\ digits.=Rimuove le cifre. Presets=Profili -Check\ Proxy\ Setting=Controlla Impostazioni Proxy -Check\ connection=Controlla la connessione -Connection\ failed\!=Connessione fallita\! -Connection\ successful\!=Connessione riuscita\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Genera gruppi da parole chiave nel campo seguente Generate\ groups\ for\ author\ last\ names=Genera gruppi in base al cognome dell'autore Regular\ expression=Espressione regolare @@ -2359,7 +2373,6 @@ Entry\ Type=Tipo di voce Entry\ types=Tipi di voce Field\ names=Nome del campo Others=Altri -Overwrite\ existing\ field\ values=Sovrascrivi i valori esistenti del campo Recommended=Consigliato Authors\ and\ Title=Autori e Titolo @@ -2427,3 +2440,23 @@ Search\ results\ from\ open\ libraries=Risultati della ricerca da librerie apert Custom\ API\ key=Chiave API personalizzata Check\ %0\ API\ Key\ Setting=Controlla le impostazioni della chiave API %0 + +Edit\ content=Modifica il contenuto +Copy\ or\ Move\ content=Copia o Sposta il contenuto +Overwrite\ field\ content=Sovrascrivi il contenuto del campo +Set=Imposta +Append=Accoda +Clear\ field\ content=Cancella il contenuto del campo +Set\ or\ append\ content=Imposta o aggiungi il contenuto +Edit\ field\ content\ for\ selected\ entries=Modifica il contenuto del campo per le voci selezionate +Rename=Rinomina +New\ field\ name=Nuovo nome del campo +Copy\ content=Copia il contenuto +Move\ content=Sposta il contenuto +Swap\ content=Scambia il contenuto +Copy\ or\ move\ the\ content\ of\ one\ field\ to\ another=Copia o sposta il contenuto di un campo in un altro +Automatic\ field\ editor=Editor automatico dei campi + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Nota\: se le voci originali mancano di parole chiave per qualificarsi per la nuova configurazione di gruppo, confermando qui le aggiungeranno) +Assign=Assegna +Do\ not\ assign=Non assegnare diff --git a/src/main/resources/l10n/JabRef_ja.properties b/src/main/resources/l10n/JabRef_ja.properties index d250ae84767..580a657d479 100644 --- a/src/main/resources/l10n/JabRef_ja.properties +++ b/src/main/resources/l10n/JabRef_ja.properties @@ -20,6 +20,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 項目 + %0\ matches\ the\ regular\ expression\ %1=%0は正規表現%1に一致します %0\ matches\ the\ term\ %1=%0は項目%1に一致します @@ -150,8 +151,6 @@ Cite\ command=Citeコマンド Clear=消去 -Clear\ fields=フィールドを消去 - Open\ /\ close\ entry\ editor=項目エディタを開閉する Close\ dialog=ダイアログを閉じる @@ -695,11 +694,18 @@ Remove\ selected\ entries\ from\ this\ group=このグループから選択項 Remove\ group=グループを削除 -Remove\ group\ "%0"?=グループ「%0」を削除しますか? +Remove\ group\ and\ subgroups=グループと下層グループを削除 + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=グループ「%0」とその下層グループを削除しますか? -Remove\ group\ and\ subgroups=グループと下層グループを削除 +Removed\ group\ "%0"\ and\ its\ subgroups.=グループ「%0」とその下層グループを削除しました. + + Remove\ link=リンクを削除 @@ -709,8 +715,6 @@ Remove\ string\ %0=文字列 %0 を削除 Removed\ group\ "%0".=グループ「%0」を削除しました. -Removed\ group\ "%0"\ and\ its\ subgroups.=グループ「%0」とその下層グループを削除しました. - Removed\ string=文字列を削除しました Renamed\ string=文字列を改名しました @@ -779,9 +783,6 @@ Select\ file\ from\ ZIP-archive=ZIP書庫からファイルを選択してくだ Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=ツリーノードを選択して表示させ,変更を受諾ないし拒否してください -Set\ field=フィールドを設定 -Set\ fields=フィールドを設定 - Settings=設定 Shortcut=ショートカット @@ -814,7 +815,6 @@ Size=サイズ Skipped\ -\ No\ PDF\ linked=跳ばしました - PDFがリンクされていません Skipped\ -\ PDF\ does\ not\ exist=跳ばしました - PDFが存在しません -Skipped\ entry.=項目を跳ばしました. Sort\ subgroups=サブグループのソート @@ -960,10 +960,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=%0行目\: No\ full\ text\ document\ found=文書本体が見つかりません Download\ from\ URL=URLからダウンロード Rename\ field=フィールドを改名しました -Append\ field=フィールドを追加 -Append\ to\ fields=フィールドに追加 -Rename\ field\ to=フィールド名を以下に変更 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=フィールドの内容を別名のフィールドに移動する Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=リモート操作用に%0ポートを使用することができません.別のアプリケーションが使用している可能性があります.別のポートを指定してみてください. @@ -1219,6 +1215,11 @@ Use\ custom\ proxy\ configuration=ユーザー定義のプロキシ設定を用 Proxy\ requires\ authentication=プロキシには認証が必要 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=警告:パスワードは平文で保管されます! Clear\ connection\ settings=接続設定を消去する +Check\ Proxy\ Setting=プロキシ設定を確認 +Check\ connection=接続を確認 +Connection\ failed\!=接続に失敗しました! +Connection\ successful\!=接続に成功しました ! + Open\ folder=フォルダを開く Export\ sort\ order=書き出し整序順 @@ -1668,7 +1669,6 @@ Protected\ terms\ file=予約語ファイル Style\ file=スタイルファイル Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice接続を開く -You\ must\ enter\ at\ least\ one\ field\ name=少なくとも一つのフィールド名を入力する必要があります Non-ASCII\ encoded\ character\ found=ASCIIエンコードでない文字が検出されました Non-UTF-8\ encoded\ field\ found=UTF-8でエンコードされていないフィールドが見つかりました Toggle\ web\ search\ interface=ウェブ検索インタフェースを入切 @@ -1806,7 +1806,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=フィールド Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' から項目データを取得することができませんでした. Entry\ from\ %0\ could\ not\ be\ parsed.=%0 からの項目を解析することができませんでした. Invalid\ identifier\:\ '%0'.=「%0」は識別子として無効です. -This\ paper\ has\ been\ withdrawn.=この論文は撤回されました. empty\ citation\ key=空の文献キー Aux\ file=auxファイル Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=特定のTeXファイルでの引用項目を集めたグループ @@ -2141,10 +2140,6 @@ Removes\ digits.=数字を削除します。 Presets=プリセット -Check\ Proxy\ Setting=プロキシ設定を確認 -Check\ connection=接続を確認 -Connection\ failed\!=接続に失敗しました! -Connection\ successful\!=接続に成功しました ! Generate\ groups\ from\ keywords\ in\ the\ following\ field=次のフィールドのキーワードからグループを生成する Generate\ groups\ for\ author\ last\ names=著者の姓でグループを生成する Regular\ expression=正規表現 @@ -2190,7 +2185,6 @@ Entry\ Type=項目型 Entry\ types=項目型 Field\ names=フィールド名\: Others=その他 -Overwrite\ existing\ field\ values=既存のフィールド値を上書き Recommended=推奨 Authors\ and\ Title=著者とタイトル @@ -2247,3 +2241,5 @@ Search\ results\ from\ open\ libraries=オープンライブラリからの検 + + diff --git a/src/main/resources/l10n/JabRef_ko.properties b/src/main/resources/l10n/JabRef_ko.properties index 2938d9c870e..90f720b7e6f 100644 --- a/src/main/resources/l10n/JabRef_ko.properties +++ b/src/main/resources/l10n/JabRef_ko.properties @@ -20,6 +20,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 항목 + %0\ matches\ the\ regular\ expression\ %1=%0은 정규식 %1와 일치합니다. %0\ matches\ the\ term\ %1=%0은 항목 %1 와 일치합니다. @@ -150,8 +151,6 @@ Cite\ command=인용 명령 Clear=초기화 -Clear\ fields=필드 초기화 - Open\ /\ close\ entry\ editor=항목 편집기 열기/닫기 Close\ dialog=대화상자 닫기 @@ -693,11 +692,18 @@ Remove\ selected\ entries\ from\ this\ group=이 그룹에서 선택한 항목 Remove\ group=그룹 삭제 -Remove\ group\ "%0"?=그룹 "%0"을 삭제하겠습니까? +Remove\ group\ and\ subgroups=그룹 및 하위 그룹 제거 + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=그룹 "%0" 및 해당 하위 그룹을 제거하시겠습니까? -Remove\ group\ and\ subgroups=그룹 및 하위 그룹 제거 +Removed\ group\ "%0"\ and\ its\ subgroups.=그룹 "%0" 및 해당 하위 그룹을 제거했습니다. + + Remove\ link=링크 삭제 @@ -707,8 +713,6 @@ Remove\ string\ %0=문자열 %0 삭제 Removed\ group\ "%0".=그룹 %0 삭제 -Removed\ group\ "%0"\ and\ its\ subgroups.=그룹 "%0" 및 해당 하위 그룹을 제거했습니다. - Removed\ string=문자열 삭제 완료 Renamed\ string=문자열 이름 변경 완료 @@ -776,9 +780,6 @@ Select\ file\ from\ ZIP-archive=ZIP 아카이브에서 파일 선택 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=변경 사항을 보고 수락하거나 거부할 트리 노드를 선택합니다. -Set\ field=필드 설정 -Set\ fields=필드 설정 - Settings=설정 Shortcut=단축키 @@ -811,7 +812,6 @@ Size=크기 Skipped\ -\ No\ PDF\ linked=건너뜀 - 연결된 PDF 없음 Skipped\ -\ PDF\ does\ not\ exist=건너뜀 - PDF가 존재하지 않습니다 -Skipped\ entry.=항목을 건너뛰었습니다. Sort\ subgroups=하위 그룹 정렬 @@ -955,10 +955,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=%0행\: 손 No\ full\ text\ document\ found=전체 텍스트 문서를 찾을 수 없습니다. Download\ from\ URL=URL에서 다운로드 Rename\ field=필드 이름 변경 -Append\ field=필드 추가 -Append\ to\ fields=필드에 추가 -Rename\ field\ to=필드 이름 변경 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=필드 내용을 다른 이름의 필드로 이동 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=원격 작업에 포트 %0을 사용할 수 없습니다. 다른 응용 프로그램에서 사용 중일 수 있습니다. 다른 포트를 지정해 보십시오. @@ -1214,6 +1210,11 @@ Use\ custom\ proxy\ configuration=사용자 지정 프록시 구성 사용 Proxy\ requires\ authentication=프록시에 인증이 필요합니다. Attention\:\ Password\ is\ stored\ in\ plain\ text\!=주의\: 비밀번호는 일반 텍스트로 저장됩니다\! Clear\ connection\ settings=연결 설정 지우기 +Check\ Proxy\ Setting=프록시 설정 확인 +Check\ connection=연결 확인 +Connection\ failed\!=연결 실패\! +Connection\ successful\!=연결 성공\! + Open\ folder=폴더 열기 Export\ sort\ order=정렬 순서 내보내기 @@ -1663,7 +1664,6 @@ Protected\ terms\ file=보호된 용어 파일 Style\ file=파일 스타일 Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice 연결 열기 -You\ must\ enter\ at\ least\ one\ field\ name=하나 이상의 필드 이름을 입력해야 합니다. Non-ASCII\ encoded\ character\ found=ASCII가 아닌 인코딩된 문자를 찾았습니다. Non-UTF-8\ encoded\ field\ found=UTF-8이 아닌 인코딩 필드를 찾았습니다. Toggle\ web\ search\ interface=웹 검색 인터페이스 전환 @@ -1801,7 +1801,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=필드 내용 Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0'에서 항목 데이터를 검색할 수 없습니다. Entry\ from\ %0\ could\ not\ be\ parsed.=%0의 항목을 구문 분석할 수 없습니다. Invalid\ identifier\:\ '%0'.=잘못된 식별자\: '%0'. -This\ paper\ has\ been\ withdrawn.=이 논문은 철회되었습니다. empty\ citation\ key=빈 인용 키 Aux\ file=AUX 파일 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=주어진 TeX 파일에서 인용된 항목을 포함하는 그룹 @@ -1905,7 +1904,7 @@ Add\ new\ String=문자열 추가 Must\ not\ be\ empty\!=비워둘 수 없습니다\! Open\ Help\ page=도움말 열기 Add\ new\ field\ name=새 필드 이름 추가 -Field\ name\:=필드 이름\: +Field\ name\:=필드 이름\: Field\ name\ "%0"\ already\ exists=필드 이름 "%0"이 이미 존재합니다 No\ field\ name\ selected\!=필드 이름을 선택하지 않았습니다 Remove\ field\ name=필드 이름 제거 @@ -2280,10 +2279,6 @@ Removes\ digits.=숫자 제거 Presets=프리셋 -Check\ Proxy\ Setting=프록시 설정 확인 -Check\ connection=연결 확인 -Connection\ failed\!=연결 실패\! -Connection\ successful\!=연결 성공\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=다음 필드의 키워드에서 그룹 생성 Generate\ groups\ for\ author\ last\ names=작가의 성씨를 가지고 그룹 생성 Regular\ expression=정규식 @@ -2340,7 +2335,6 @@ Entry\ Type=목록 유형 Entry\ types=목록 유형들 Field\ names=필드 이름 Others=기타 -Overwrite\ existing\ field\ values=기존의 필드 값 덮어쓰기 Recommended=권장 Authors\ and\ Title=작가와 제목 @@ -2401,3 +2395,5 @@ Keep\ dialog\ always\ on\ top=대화창 위로 고정 + + diff --git a/src/main/resources/l10n/JabRef_nl.properties b/src/main/resources/l10n/JabRef_nl.properties index 6b183bbe7f9..312e3c7e358 100644 --- a/src/main/resources/l10n/JabRef_nl.properties +++ b/src/main/resources/l10n/JabRef_nl.properties @@ -1,3 +1,4 @@ +Could\ not\ delete\ empty\ entries.=Kan lege invoer niet verwijderen. @@ -13,6 +14,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 komt overeen met de standaard-uitdruk %1 %0\ matches\ the\ term\ %1=%0 komt overeen met de term %1 @@ -135,8 +137,6 @@ Cite\ command=Citeer opdracht Clear=Wissen -Clear\ fields=Velden wissen - Close\ dialog=Sluit dialoog @@ -638,11 +638,18 @@ Remove\ selected\ entries\ from\ this\ group=Geselecteerde items uit deze groep Remove\ group=Verwijder groep -Remove\ group\ "%0"?=Verwijder groep "%0"? +Remove\ group\ and\ subgroups=Verwijder groep en subgroepen + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Verwijder groep "%0" en zijn subgroepen? -Remove\ group\ and\ subgroups=Verwijder groep en subgroepen +Removed\ group\ "%0"\ and\ its\ subgroups.=Groep "%0" en zijn subgroepen verwijderd. + + Remove\ link=Verwijder link @@ -651,8 +658,6 @@ Remove\ old\ entry=Verwijder oude invoer Removed\ group\ "%0".=Groep "%0" verwijderd. -Removed\ group\ "%0"\ and\ its\ subgroups.=Groep "%0" en zijn subgroepen verwijderd. - Removed\ string=Tekenreeks verwijderd Renamed\ string=Tekenreeks hernoemd @@ -719,9 +724,6 @@ Select\ file\ from\ ZIP-archive=Selecteer bestand van ZIP-archief Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecteer de boom knopen om veranderingen te tonen en te accepteren of afwijzen -Set\ field=Veld instellen -Set\ fields=Velden instellen - Settings=Instellingen Shortcut=Snelkoppeling @@ -754,7 +756,6 @@ Size=Grootte Skipped\ -\ No\ PDF\ linked=Overgeslagen - Geen PDF gelinkt Skipped\ -\ PDF\ does\ not\ exist=Overgeslagen - PDF bestaat niet -Skipped\ entry.=Overgeslagen invoer. source\ edit=broncode aanpassen @@ -872,10 +873,6 @@ MIME\ type=MIME type No\ full\ text\ document\ found=Geen volledig tekstdocument gevonden Download\ from\ URL=Download vanaf URL Rename\ field=Veld hernoemen -Append\ field=Veld toevoegen -Append\ to\ fields=Toevoegen aan velden -Rename\ field\ to=Veld hernoemen naar -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Verplaats de inhoud van een veld naar een veld met een andere naam Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan poort %0 niet gebruiken voor bediening op afstand; een andere applicatie heeft deze mogelijk in gebruik. Probeer een andere poort op te geven. @@ -1090,6 +1087,7 @@ Proxy\ requires\ authentication=Proxy vereist authenticatie Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Opgelet\: Wachtwoord wordt opgeslagen in platte tekst\! Clear\ connection\ settings=Verbindingsinstellingen wissen + Open\ folder=Map openen Export\ sort\ order=Sorteervolgorde exporteren @@ -1494,7 +1492,6 @@ Protected\ terms\ file=Beschermde voorwaarden bestand Style\ file=Bestandsstijl Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice verbinding openen -You\ must\ enter\ at\ least\ one\ field\ name=U moet ten minste één veldnaam opgeven Non-ASCII\ encoded\ character\ found=Niet-ASCII gecodeerd teken gevonden Toggle\ web\ search\ interface=Web zoek interface in-/uitschakelen @@ -1615,7 +1612,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Verwijdert alle Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Kon geen data ophalen van '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Invoer van %0 kon niet worden geparseerd. Invalid\ identifier\:\ '%0'.=Ongeldig Id\: '%0'. -This\ paper\ has\ been\ withdrawn.=Dit artikel is ingetrokken. Aux\ file=Aux bestand Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Groep bevat invoergegevens aangehaald in een bepaald TeX-bestand @@ -1763,6 +1759,8 @@ Default\ pattern=Standaard patroon + + diff --git a/src/main/resources/l10n/JabRef_no.properties b/src/main/resources/l10n/JabRef_no.properties index 164dcbe8bb2..d8de347be33 100644 --- a/src/main/resources/l10n/JabRef_no.properties +++ b/src/main/resources/l10n/JabRef_no.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 matcher regulæruttrykket %1 %0\ matches\ the\ term\ %1=%0 matcher uttrykket %1 @@ -126,8 +127,6 @@ Cite\ command=Siteringskommando Clear=Opphev -Clear\ fields=Slett felter - Close\ dialog=Lukk dialog @@ -604,11 +603,18 @@ Remove\ selected\ entries\ from\ this\ group=Fjern valgte oppføringer fra denne Remove\ group=Fjern gruppe -Remove\ group\ "%0"?=Fjern gruppen "%0"? +Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Fjern gruppen "%0" og dens undergrupper? -Remove\ group\ and\ subgroups=Fjern gruppe og undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernet gruppen "%0" og dens undergrupper + + Remove\ link=Slett link @@ -617,8 +623,6 @@ Remove\ old\ entry=Fjern gammel enhet Removed\ group\ "%0".=Fjernet gruppen "%0" -Removed\ group\ "%0"\ and\ its\ subgroups.=Fjernet gruppen "%0" og dens undergrupper - Removed\ string=Fjernet streng Renamed\ string=Endret navn på streng @@ -681,9 +685,6 @@ Select\ file\ from\ ZIP-archive=Velg fil fra ZIP-fil Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Velg trenodene for å inspisere og akseptere eller avslå endringer -Set\ field=Sett felt -Set\ fields=Sett felter - Settings=Innstillinger Shortcut=Snarvei @@ -714,7 +715,6 @@ Size=Størrelse Skipped\ -\ No\ PDF\ linked=Hoppet over - ingen PDF-fil linket Skipped\ -\ PDF\ does\ not\ exist=Hoppet over - PDF-filen finnes ikke -Skipped\ entry.=Hoppet over enhet. source\ edit=redigering av kilde @@ -828,8 +828,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Last ned fra URL Rename\ field=Endre navn på felt -Rename\ field\ to=Endre navn på felt til -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytt innholdet i et felt til et annet felt Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan ikke bruke port %0 for fjernstyring; den kan være i bruk av et annet program. Pr¸v å spesifisere en annen port. @@ -962,6 +960,7 @@ Finished\ automatically\ setting\ external\ links.=Fullførte automatisk setting + Open\ folder=Åpne mapp Export\ sort\ order=Sorteringsrekkefølge ved eksport @@ -1203,6 +1202,8 @@ Default\ pattern=Standardmønster + + diff --git a/src/main/resources/l10n/JabRef_pl.properties b/src/main/resources/l10n/JabRef_pl.properties index 153fa9456a9..e8a82418acc 100644 --- a/src/main/resources/l10n/JabRef_pl.properties +++ b/src/main/resources/l10n/JabRef_pl.properties @@ -13,6 +13,7 @@ + Abbreviation=Skrót About\ JabRef=O JabRef @@ -95,8 +96,6 @@ Change\ entry\ type=Zmień typ wpisu Clear=Wyczyść -Clear\ fields=Wyczyść pola - Close\ dialog=Zamknij okno @@ -472,9 +471,16 @@ Remove\ subgroups=Usuń podgrupy Remove\ group=Usuń grupę +Remove\ group\ and\ subgroups=Usuń grupę i podgrupy + + + + + + + -Remove\ group\ and\ subgroups=Usuń grupę i podgrupy Remove\ link=Usuń łącze @@ -484,7 +490,6 @@ Remove\ old\ entry=Usuń stary wpis - Replace=Zamień Replace\ With\:=Zamień na\: Find\:=Znajdź\: @@ -529,7 +534,6 @@ Select\ entry\ type=Wybierz typ wpisu - Settings=Ustawienia Shortcut=Skrót @@ -551,7 +555,6 @@ Simple\ HTML=Uproszczony HTML Size=Rozmiar -Skipped\ entry.=Pominięty wpis. @@ -627,7 +630,6 @@ MIME\ type=Typ MIME Download\ from\ URL=Pobierz z adresu URL Rename\ field=Zmień nazwę pola -Append\ to\ fields=Dołącz do pól Please\ enter\ a\ valid\ file\ path.=Wprowadź prawidłową ścieżkę do pliku. @@ -717,6 +719,7 @@ Hostname=Nazwa hosta Proxy\ configuration=Konfiguracja proxy + Open\ folder=Otwórz folder Parsing\ error=Błąd parsowania @@ -1077,6 +1080,8 @@ Reset=Reset + + diff --git a/src/main/resources/l10n/JabRef_pt.properties b/src/main/resources/l10n/JabRef_pt.properties index a4cf86dab14..6fd581ffae8 100644 --- a/src/main/resources/l10n/JabRef_pt.properties +++ b/src/main/resources/l10n/JabRef_pt.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 encontrada a expressão regular %1 %0\ matches\ the\ term\ %1=%0 encontrado o termo %1 @@ -130,8 +131,6 @@ Cite\ command=Comando citar Clear=Limpar -Clear\ fields=Limpar campos - Close\ dialog=Fechar janela de diálogo @@ -629,11 +628,18 @@ Remove\ entry\ from\ import=Remover referência da importação Remove\ group=Remover grupo -Remove\ group\ "%0"?=Remover grupo "%0"? +Remove\ group\ and\ subgroups=Remover grupos e subgrupos + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Remover grupo "%0" e seus subgrupos? -Remove\ group\ and\ subgroups=Remover grupos e subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. + + Remove\ link=Remover link @@ -642,8 +648,6 @@ Remove\ old\ entry=Remover referência antiga Removed\ group\ "%0".=Grupo "%0" removido. -Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. - Removed\ string=String removida Renamed\ string=String renomeada @@ -696,9 +700,6 @@ Select\ file\ from\ ZIP-archive=Selecionar arquivo a partir de um arquivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecione os nós da árvore para visualizar e aceitar ou rejeitar mudanças -Set\ field=Configurar campo -Set\ fields=Configurar campos - Settings=Configurações Shortcut=Atalho @@ -729,7 +730,6 @@ Size=Tamanho Skipped\ -\ No\ PDF\ linked=Omitido - Nenhum PDF linkado Skipped\ -\ PDF\ does\ not\ exist=Omitido - O PDF não existe -Skipped\ entry.=Referência omitida. source\ edit=edição de fonte @@ -840,8 +840,6 @@ Unable\ to\ open\ link.=Não foi possível abrir link. This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ running\ instance\ of\ JabRef\ instead\ of\ opening\ a\ new\ instance.\ For\ instance,\ this\ is\ useful\ when\ you\ open\ a\ file\ in\ JabRef\ from\ your\ web\ browser.\ Note\ that\ this\ will\ prevent\ you\ from\ running\ more\ than\ one\ instance\ of\ JabRef\ at\ a\ time.=Esta funcionalidade permite que novos arquivos sejam abertos ou importados para uma instância do JabRef já aberta ao invés de abrir uma nova instância. Por exemplo, isto é útil quando você abre um arquivo no JabRef a partir de ser navegador web. Note que isto irá previnir que você execute uma ou mais instâncias do JabRef ao mesmo tempo. Rename\ field=Renomear campo -Rename\ field\ to=Renomear campo para -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover conteúdo de um campo para um campo com nome diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Não é possível utilizar a porta %0 para operação remota; outra aplicação pode estar usando-a. Tente utilizar uma outra porta. @@ -1050,6 +1048,7 @@ Use\ custom\ proxy\ configuration=Usar configurações personalizadas de proxy Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atenção\: A senha é armazenada em texto simples\! Clear\ connection\ settings=Limpar configurações da conexão + Open\ folder=Abrir diretório Export\ sort\ order=Exportar ordenação @@ -1355,6 +1354,8 @@ Default\ pattern=Ppadrão predefinido + + diff --git a/src/main/resources/l10n/JabRef_pt_BR.properties b/src/main/resources/l10n/JabRef_pt_BR.properties index f75024aafc7..179f7dd8128 100644 --- a/src/main/resources/l10n/JabRef_pt_BR.properties +++ b/src/main/resources/l10n/JabRef_pt_BR.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 entradas -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 exportação bem-sucedida. Abrir a pasta que contém o arquivo salvo? + %0\ matches\ the\ regular\ expression\ %1=%0 encontrada a expressão regular %1 @@ -151,8 +151,6 @@ Cite\ command=Comando citar Clear=Limpar -Clear\ fields=Limpar campos - Open\ /\ close\ entry\ editor=Abrir / fechar o editor de referências Close\ dialog=Fechar janela de diálogo @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Remover as referências selecionada Remove\ group=Remover grupo -Remove\ group\ "%0"?=Remover grupo "%0"? +Remove\ group\ and\ subgroups=Remover grupos e subgrupos + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Remover grupo "%0" e seus subgrupos? -Remove\ group\ and\ subgroups=Remover grupos e subgrupos +Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. + + Remove\ link=Remover link @@ -717,8 +722,6 @@ Remove\ string\ %0=Remover string %0 Removed\ group\ "%0".=Grupo "%0" removido. -Removed\ group\ "%0"\ and\ its\ subgroups.=Grupo "%0" e seus subgrupos removidos. - Removed\ string=String removida Renamed\ string=String renomeada @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=Selecionar arquivo a partir de um arquivo ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Selecione os nós da árvore para visualizar e aceitar ou rejeitar mudanças -Set\ field=Configurar campo -Set\ fields=Configurar campos - Settings=Configurações Shortcut=Atalho @@ -825,7 +825,6 @@ Size=Tamanho Skipped\ -\ No\ PDF\ linked=Omitido - Nenhum PDF linkado Skipped\ -\ PDF\ does\ not\ exist=Omitido - O PDF não existe -Skipped\ entry.=Referência omitida. Sort\ subgroups=Ordenar subgrupos @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Linha %0\: En No\ full\ text\ document\ found=Nenhum documento encontrado Download\ from\ URL=Baixar da URL Rename\ field=Renomear campo -Append\ field=Acrescentar campo -Append\ to\ fields=Acrescentar aos campos -Rename\ field\ to=Renomear campo para -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Mover conteúdo de um campo para um campo com nome diferente Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Não é possível utilizar a porta %0 para operação remota; outra aplicação pode estar usando-a. Tente utilizar uma outra porta. @@ -1242,6 +1237,11 @@ Use\ custom\ proxy\ configuration=Usar configurações personalizadas de proxy Proxy\ requires\ authentication=Proxy requer autenticação Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Atenção\: A senha é armazenada em texto simples\! Clear\ connection\ settings=Limpar configurações da conexão +Check\ Proxy\ Setting=Verificar Configuração de Proxy +Check\ connection=Verificar conexão +Connection\ failed\!=Falha na conexão\! +Connection\ successful\!=Conexão bem-sucedida\! + Open\ folder=Abrir diretório Export\ sort\ order=Exportar ordenação @@ -1694,7 +1694,6 @@ Protected\ terms\ file=Arquivo de termos protegidos Style\ file=Arquivo de estilo Open\ OpenOffice/LibreOffice\ connection=Abrir conexão com OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Você deve digitar pelo menos um nome de campo Non-ASCII\ encoded\ character\ found=Caractere não-ASCII encontrado Non-UTF-8\ encoded\ field\ found=Campo não-UTF-8 encontrado Toggle\ web\ search\ interface=Alternar interface de busca na web @@ -1832,7 +1831,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Remove todas as Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Não foi possível recuperar dados de referência a partir de '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Não foi possível analisar a referência a partir de %0. Invalid\ identifier\:\ '%0'.=Identificador inválido\: '%0'. -This\ paper\ has\ been\ withdrawn.=Este documento foi retirado. empty\ citation\ key=chave de citação vazia Aux\ file=Arquivo Aux Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo contendo referências citadas em um arquivo TeX @@ -2314,10 +2312,6 @@ Removes\ digits.=Remove dígitos. Presets=Predefinições -Check\ Proxy\ Setting=Verificar Configuração de Proxy -Check\ connection=Verificar conexão -Connection\ failed\!=Falha na conexão\! -Connection\ successful\!=Conexão bem-sucedida\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Gerar grupos de palavras-chave no campo seguinte Generate\ groups\ for\ author\ last\ names=Gerar grupos para os últimos nomes dos autores Regular\ expression=Expressão regular @@ -2375,7 +2369,6 @@ Entry\ Type=Tipo de referência Entry\ types=Tipos de referência Field\ names=Nomes dos campos Others=Outros -Overwrite\ existing\ field\ values=Sobrescrever valores de campos existentes Recommended=Recomendado Authors\ and\ Title=Autores e Título @@ -2439,3 +2432,5 @@ Search\ results\ from\ open\ libraries=Resultados da pesquisa para as biblioteca + + diff --git a/src/main/resources/l10n/JabRef_ru.properties b/src/main/resources/l10n/JabRef_ru.properties index 4f2f56445c7..0999c1ddcb0 100644 --- a/src/main/resources/l10n/JabRef_ru.properties +++ b/src/main/resources/l10n/JabRef_ru.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 записей +Export\ operation\ finished\ successfully.=Операция экспорта выполнена успешно. + +Reveal\ in\ File\ Explorer=Открыть в Проводнике %0\ matches\ the\ regular\ expression\ %1=%0 соответствует регулярному выражению %1 @@ -150,8 +153,6 @@ Cite\ command=Команда цитирования Clear=Очистить -Clear\ fields=Очистить поля - Open\ /\ close\ entry\ editor=Открыть / закрыть редактор записей Close\ dialog=Закрыть диалоговое окно @@ -173,6 +174,8 @@ Copy=Копировать Copy\ title=Копировать заглавие Copy\ \\cite{citation\ key}=Копировать \\ цитировать{citation key} +Copy\ citation\ (html)=Копировать цитирование (html) +Copy\ citation\ (text)=Копировать цитирование (текст) Copy\ citation\ key=Копировать ключ цитаты Copy\ citation\ key\ and\ link=Скопировать ключ цитаты и ссылку Copy\ citation\ key\ and\ title=Копировать ключ цитаты и заголовок @@ -230,6 +233,8 @@ Library\ properties=Свойства БД Default=По умолчанию +Character\ encoding\ UTF-8\ is\ not\ supported.=Кодировка UTF-8 не поддерживается. +UTF-8\ could\ not\ be\ used\ to\ encode\ the\ following\ characters\:\ %0=Не удалось использовать UTF-8 для кодирования следующих символов\: %0 The\ chosen\ encoding\ '%0'\ could\ not\ encode\ the\ following\ characters\:=В выбранной кодировке '%0' невозможно прочитать следующие символы\: Downloading=Загрузка … @@ -278,6 +283,7 @@ Downloaded\ website\ as\ an\ HTML\ file.=Загружен сайт в форма duplicate\ removal=удаление дубликатов +Duplicate\ fields=Дубликаты полей Duplicate\ string\ name=Дубликат имени строки @@ -536,7 +542,9 @@ Moved\ group\ "%0".=Перемещенная группа "%0". Mr.\ DLib\ Privacy\ settings=Настройки конфиденциальности сервиса "Mr. DLib" +No\ database\ is\ open=База данных не открыта +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=Нам требуется база данных для экспорта. Откройте её. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=Веб сервис "Mr. DLib" не предлагает рекомендаций для этой записи. @@ -695,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=Удалить выбранные Remove\ group=Удалить группу -Remove\ group\ "%0"?=Удалить группу "%0"? +Remove\ group\ and\ subgroups=Удалить группу и подгруппы + +Remove\ groups\ and\ subgroups=Удалить группы и подгруппы + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=Удалить все выбранные группы, сохранив их подгруппы? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=Удалить группу "%0", сохранив её подгруппы? + +Remove\ groups=Удалить группы + +Removed\ all\ selected\ groups.=Все выбранные группы удалены. Remove\ group\ "%0"\ and\ its\ subgroups?=Удалить группу "%0" и ее подгруппы? -Remove\ group\ and\ subgroups=Удалить группу и подгруппы +Removed\ group\ "%0"\ and\ its\ subgroups.=Группа "%0" и ее подгруппы удалены. + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=Удалить все выбранные группы и их подгруппы? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=Все выбранные группы и их подгруппы удалены. Remove\ link=Удалить ссылку @@ -709,8 +731,6 @@ Remove\ string\ %0=Удалить строку %0 Removed\ group\ "%0".=Группа "%0" удалена. -Removed\ group\ "%0"\ and\ its\ subgroups.=Группа "%0" и ее подгруппы удалены. - Removed\ string=Строка удалена Renamed\ string=Строка переименована @@ -733,6 +753,8 @@ Replaces\ Unicode\ ligatures\ with\ their\ expanded\ form=Заменяет ли Required\ fields=Обязательные поля +Do\ not\ resolve\ BibTeX\ strings=Не разрешать строки BibTeX +Resolve\ BibTeX\ strings\ for\ the\ following\ fields=Разрешить строки BibTeX для следующих полей resolved=разрешено @@ -740,11 +762,13 @@ Restart=Перезапустить Restart\ required=Требуется перезапуск +Return\ to\ dialog=Вернуться в диалоговое окно Review=Просмотр Review\ changes=Просмотр изменений Review\ Field\ Migration=Перемещение поля обзора +Loading=Загрузка Save=Сохранить Save\ all\ finished.=Сохранить все завершенные. @@ -778,9 +802,6 @@ Select\ file\ from\ ZIP-archive=Выбрать файл из ZIP-архива Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Выбрать узлы дерева для просмотра и применения/отклонения изменений -Set\ field=Настройка поля -Set\ fields=Настройка полей - Settings=Параметры Shortcut=Ярлык @@ -813,7 +834,12 @@ Size=Размер Skipped\ -\ No\ PDF\ linked=Пропущено - Без ссылок на PDF Skipped\ -\ PDF\ does\ not\ exist=Пропущено - PDF не существует -Skipped\ entry.=Запись пропущена. +JabRef\ skipped\ the\ entry.=JabRef пропустил запись. +Import\ error=Ошибка импортирования +Open\ library\ error=Ошибка при открытии библиотеки +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=Пожалуйста, проверьте ваш файл библиотеки на правильность синтаксиса. +SourceTab\ error=Ошибка SourceTab +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=Пользовательский ввод посредством редактора записей во вкладке `{}bibtex source` привёл к возникновению ошибки. Sort\ subgroups=Сортировать подгруппы @@ -902,6 +928,7 @@ Warning=Предупреждение Warnings=Предупреждения +Warning\:\ You\ added\ field\ "%0"\ twice.\ Only\ one\ will\ be\ kept.=Внимание\: Вы добавили поле "%0" дважды. Только одно из них будет сохранено. web\ link=веб-ссылки @@ -959,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Строка No\ full\ text\ document\ found=Не найдено полнотекстовых документов Download\ from\ URL=Загрузить из URL-адреса Rename\ field=Переименовать поле -Append\ field=Добавить поле -Append\ to\ fields=Добавить к полям -Rename\ field\ to=Переименовать поле в -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Переместить содержимое поля в поле с другим именем Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Невозможно использовать порт %0 для удаленного подключения; another application may be using it. Try specifying another port. @@ -1032,6 +1055,7 @@ Cite\ special=Особое цитирование Extra\ information\ (e.g.\ page\ number)=Дополнительные сведения (напр., номер страницы) Manage\ citations=Управление цитированием Problem\ modifying\ citation=Ошибка при модификации цитаты +Problem\ collecting\ citations=Возникла проблема при сборе ссылок Citation=Цитата Connecting...=Подключение... Select\ style=Выбор стиля @@ -1046,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=Синхронизация библи Select\ which\ open\ Writer\ document\ to\ work\ on=Выберите открытый документ OpenOffice/LibreOffice Writer для работы Connected\ to\ document=Подключено к документу +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Не удалось выполнить подключение к документу Writer. Убедитесь, что документ открыт перед использованием кнопки 'Выбрать документ Writer', чтобы выполнить подключение. +No\ Writer\ documents\ found=Документы Writer не найдены Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=Вставка цитаты без текста (запись отображается в списке ссылок) Cite\ selected\ entries\ with\ extra\ information=Цитировать выбранные записи с дополнительными сведениями @@ -1054,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=Проверьте актуал Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=Ваш документ OpenOffice/LibreOffice ссылается на ключ цитирования '%0', который не найден в вашей текущей библиотеке. +This\ operation\ requires\ a\ bibliography\ database.=Этой операции требуется библиографическая база данных. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Ссылки в вашем документе OpenOffice/LibreOffice содержат не менее %0 ключей цитирования, которые не удалось найти в текущей библиотеке. Некоторые из них %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=В вашем документе OpenOffice/LibreOffice отсутствуют какие-либо ключи цитирования, которые также можно найти в вашей текущей библиотеке. Unable\ to\ synchronize\ bibliography=Не удалось выполнить синхронизацию библиографии Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=Объединить пары цитат, разделенные только пробелом @@ -1063,7 +1092,12 @@ Autodetection\ failed=Ошибка автоопределения Please\ wait...=Подождите... Connection\ lost=Соединение прервано +Could\ not\ update\ bibliography=Не удалось обновить библиографию +Not\ connected\ to\ document=Нет подключения к документу +Problem\ combining\ cite\ markers=Проблема при объединении отметок цитирования +Unable\ to\ reload\ style\ file=Не удалось перезагрузить файл оформления +Problem\ during\ separating\ cite\ markers=Проблема при разделении отметок цитирования Automatically\ sync\ bibliography\ when\ inserting\ citations=Автоматически синхронизировать библиографию при вставке цитат Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=Поиск записей BibTeX только в активной вкладке @@ -1083,6 +1117,7 @@ Style\ selection=Выбор стиля No\ valid\ style\ file\ defined=Не определен допустимый файл стиля Choose\ pattern=Выбор шаблона Search\ and\ store\ files\ relative\ to\ library\ file\ location=Поиск и хранение файлов относительно файла библиотеки +File\ directory=Каталог файлов Could\ not\ run\ the\ gnuclient/emacsclient\ program.\ Make\ sure\ you\ have\ the\ emacsclient/gnuclient\ program\ installed\ and\ available\ in\ the\ PATH.=Не удалось запустить приложение gnuclient/emacsclient. Убедитесь, что приложение gnuclient/emacsclient установлено и доступно по указанному пути PATH. You\ must\ select\ either\ a\ valid\ style\ file,\ or\ use\ one\ of\ the\ default\ styles.=Необходимо использовать допустимый файл стиля или один из стилей по умолчанию. @@ -1165,6 +1200,7 @@ Toggle\ relevance=Показать/скрыть релевантность Toggle\ quality\ assured=Прказать/скрыть проверку контроля качества Toggle\ print\ status=Изменить статус 'печать' Update\ keywords=Изменить ключевые слова +Problem\ connecting=Проблема при подключении Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=Соединение с OpenOffice/LibreOffice прервано. Убедитесь, что приложение OpenOffice/LibreOffice запущено и повторите соединение. JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef отправляет по крайней мере один запрос для заметки издателю. @@ -1179,6 +1215,7 @@ Removed\ all\ subgroups\ of\ group\ "%0".=Все подгруппы группы To\ disable\ the\ memory\ stick\ mode\ rename\ or\ remove\ the\ jabref.xml\ file\ in\ the\ same\ folder\ as\ JabRef.=Для отключения режима флеш-памяти переименуйте или удалите файл jabref.xml в каталоге JabRef. Unable\ to\ connect.\ One\ possible\ reason\ is\ that\ JabRef\ and\ OpenOffice/LibreOffice\ are\ not\ both\ running\ in\ either\ 32\ bit\ mode\ or\ 64\ bit\ mode.=Не удалось выполнить подключение. Возможная причина\: JabRef и OpenOffice/LibreOffice не запущены в режиме 32-х или 64-х битного приложения. Delimiter(s)=Разделитель(и) +When\ downloading\ files,\ or\ moving\ linked\ files\ to\ the\ file\ directory,\ use\ the\ bib\ file\ location.=При загрузке файлов или перемещении связанных файлов в каталог файлов используйте расположение файла bib. Searching...=Выполняется поиск... Please\ enter\ a\ search\ string=Введите строку для поиска @@ -1216,6 +1253,16 @@ Use\ custom\ proxy\ configuration=Использовать пользовате Proxy\ requires\ authentication=Для прокси-сервера требуется авторизация Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Внимание. Пароль сохранен в виде простого текста. Clear\ connection\ settings=Удалить настройки подключения +Check\ Proxy\ Setting=Проверить настройки прокси +Check\ connection=Проверить подключение +Connection\ failed\!=Соединение потеряно\! +Connection\ successful\!=Соединение успешно\! + +SSL\ Configuration=Конфигурация SSL +SSL\ configuration\ changed=Конфигурация SSL изменена +SSL\ certificate\ file=Файл сертификата SSL +Duplicate\ Certificates=Дубликаты сертификатов +You\ already\ added\ this\ certificate=Вы уже добавили этот сертификат Open\ folder=Открыть папку Export\ sort\ order=Экспорт в порядке сортировки @@ -1252,6 +1299,7 @@ Please\ open\ %0\ manually.=Откройте %0 вручную. The\ link\ has\ been\ copied\ to\ the\ clipboard.=Ссылка скопирована в буфер обмена. Open\ %0\ file=Открыть файл %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=Не удалось обнаружить терминал автоматически. Пожалуйста, укажите пользовательский терминал в настройках. Cannot\ delete\ file=Невозможно удалить файл File\ permission\ error=Ошибка прав доступа для файла @@ -1558,10 +1606,21 @@ Custom=Определено пользователем Export\ cited=Экспорт цитированного Unable\ to\ generate\ new\ library=Не удалось создать новую БД +The\ cursor\ is\ in\ a\ protected\ area.=Курсор находится в защищённой области. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=Не открыто библиографических баз данных для цитирования. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=Не открыто библиографических баз данных для обновления отметок ссылок после цитирования. +No\ bibliography\ entries\ are\ selected\ for\ citation.=Не выбраны библиографические записи для выполнения цитирования. +No\ bibliography\ style\ is\ selected\ for\ citation.=Не выбрано оформление библиографии для выполнения цитирования. +No\ database=Нет базы данных +No\ entries\ selected\ for\ citation=Не выбрано записей для выполнения цитирования +Open\ one\ before\ citing.=Откройте одну из них перед цитированием. +Select\ one\ before\ citing.=Выберите одну из них перед цитированием. +Select\ some\ before\ citing.=Выберите несколько перед выполнением цитирования. Found\ identical\ ranges=Найдены идентичные диапазоны Found\ overlapping\ ranges=Найдены пересекающиеся диапазоны @@ -1628,6 +1687,8 @@ Issue\ report\ successful=Отчёт об ошибке успешно отпра Your\ issue\ was\ reported\ in\ your\ browser.=Отчет о проблеме был отправлен в браузер. The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=Лог и информация об ошибке успешно скопированы. Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=Пожалуйста, вставьте эту информацию (с помощью Ctrl+V) в описание проблемы. +Last\ notification=Последнее уведомление +Check\ the\ event\ log\ to\ see\ all\ notifications=Проверьте журнал событий для просмотра всех уведомлений Host=Хост Port=Порт @@ -1665,7 +1726,6 @@ Protected\ terms\ file=Файл с защищенными терминами Style\ file=Файл стиля Open\ OpenOffice/LibreOffice\ connection=Установить подключение к Open OpenOffice/LibreOffice -You\ must\ enter\ at\ least\ one\ field\ name=Необходимо ввести минимум одно имя поля Non-ASCII\ encoded\ character\ found=Обнаружен символ не в кодировке ASCII Non-UTF-8\ encoded\ field\ found=Найдено поле НЕ в кодировке UTF-8 Toggle\ web\ search\ interface=Переключение интерфейса веб-поиска @@ -1803,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Удаляет Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Не удалось получить данные из записи '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Данные в %0 не могут быть обработаны. Invalid\ identifier\:\ '%0'.=Неверный идентификатор\: '%0'. -This\ paper\ has\ been\ withdrawn.=Настоящий документ был отозван. empty\ citation\ key=пустой ключ цитаты Aux\ file=Aux файл Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Группа, содержащая записи в заданном TeX файле @@ -1831,6 +1890,7 @@ Could\ not\ connect\ to\ Vim\ server.\ Make\ sure\ that\ Vim\ is\ running\ with\ Could\ not\ connect\ to\ a\ running\ gnuserv\ process.\ Make\ sure\ that\ Emacs\ or\ XEmacs\ is\ running,\ and\ that\ the\ server\ has\ been\ started\ (by\ running\ the\ command\ 'server-start'/'gnuserv-start').=Не удалось подключиться к запущенному процессу gnuserv. Убедитесь, что приложение Emacs или XEmacsи сервер запущены(с помощью команды 'server-start'/'gnuserv-start'). Error\ pushing\ entries=Ошибка при отправке записей +Preamble=Преамбула Markings=Маркировка Use\ selected\ instance=Использовать выбранный экземпляр @@ -1849,6 +1909,7 @@ Blog=Блог Check\ integrity=Проверка целостности Cleanup\ URL\ link=Очистить URL ссылку Cleanup\ URL\ link\ by\ removing\ special\ symbols\ and\ extracting\ simple\ link=Очистка URL ссылку, удалив специальные символы, чтобы получился простой URL +Copy\ DOI=Копировать DOI Copy\ DOI\ url=Копировать URL-адрес DOI Development\ version=Версия для разработчиков Export\ selected\ entries=Экспорт выбранных записей @@ -1858,6 +1919,8 @@ JabRef\ resources=Ресурсы JabRef Manage\ journal\ abbreviations=Настроить аббревиатуры журналов Manage\ protected\ terms=Управление защищенными терминами New\ entry\ from\ plain\ text=Новая запись из обычного текста +Import\ by\ ID=Импортировать по ID +Enter\ a\ valid\ ID=Введите действительный ID New\ sublibrary\ based\ on\ AUX\ file=Новая под-библиотека из файла AUX Push\ entries\ to\ external\ application\ (%0)=Передать записи во внешнее приложение (%0) Quit=Выход @@ -1917,6 +1980,7 @@ Keyword\ separator=Разделитель ключевых слов Remove\ keyword=Удалить слово Are\ you\ sure\ you\ want\ to\ remove\ keyword\:\ "%0"?=Вы уверены, что вы хотите удалить слово\: "%0"? Reset\ to\ default=Сбросить настройки +String\ constants=Строковые постоянные Export\ all\ entries=Экспорт всех записей Generate\ citation\ keys=Генерировать ключи цитирования Manage\ field\ names\ &\ content=Управлять названиями полей и контентом @@ -2046,6 +2110,7 @@ Please\ provide\ a\ valid\ aux\ file.=Пожалуйста, предоставь Keyword\ delimiter=Разделитель ключевых слов Hierarchical\ keyword\ delimiter=Разделитель иерархических ключевых слов Escape\ ampersands=Экранировка амперсандов +Escape\ dollar\ sign=Экранированный символ доллара Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Подсказка\:\nдля поиска всех полей со значением Smith, введите\:\nsmith\nДля поиска в поле Автор значения Smith а в поле Заголовок значения electrical, введите\:\nauthor\=smith and title\=electrical @@ -2234,7 +2299,7 @@ This\ entry\ type\ is\ intended\ for\ sources\ such\ as\ web\ sites\ which\ are\ A\ single-volume\ work\ of\ reference\ such\ as\ an\ encyclopedia\ or\ a\ dictionary.=Неделимая работа или ссылка, как энциклопедия или словарь. A\ technical\ report,\ research\ report,\ or\ white\ paper\ published\ by\ a\ university\ or\ some\ other\ institution.=Технический отчет, исследовательский отчет, или белая книга, выпущенная институтом или другим учреждением. An\ entry\ set\ is\ a\ group\ of\ entries\ which\ are\ cited\ as\ a\ single\ reference\ and\ listed\ as\ a\ single\ item\ in\ the\ bibliography.=Набор записей представляет собой группу записей, которые приводятся в виде единой ссылки и перечислены в виде одного элемента в библиографии. -Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. +Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. Supplemental\ material\ in\ a\ "Collection".=Дополнительные материалы в "Коллекции". Supplemental\ material\ in\ a\ "Periodical".\ This\ type\ may\ be\ useful\ when\ referring\ to\ items\ such\ as\ regular\ columns,\ obituaries,\ letters\ to\ the\ editor,\ etc.\ which\ only\ have\ a\ generic\ title.=Дополнительные материалы в "Периодическом издании". Этот тип может быть полезен при обращении к таким элементам, как обычные колонки, некрологи, письма к редактору и т.д., которые имеют только общее название. A\ thesis\ written\ for\ an\ educational\ institution\ to\ satisfy\ the\ requirements\ for\ a\ degree.=Тезис, написанный для учебного заведения с целью удовлетворения требований к степени. @@ -2280,10 +2345,6 @@ Removes\ digits.=Удаляет цифры. Presets=Предустановки -Check\ Proxy\ Setting=Проверить настройки прокси -Check\ connection=Проверить подключение -Connection\ failed\!=Соединение потеряно\! -Connection\ successful\!=Соединение успешно\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Создать группы из ключевых слов в следующем поле Generate\ groups\ for\ author\ last\ names=Создание групп для фамилий авторов Regular\ expression=Регулярное выражение @@ -2305,24 +2366,40 @@ Separate\ citations=Разделить цитаты Unprotect\ terms=Незащищенные термины Generate\ a\ new\ key\ for\ imported\ entries\ (overwriting\ their\ default)=Создать новый ключ для импортированных записей (взамен существующих) +Warn\ about\ duplicates\ on\ import=Предупреждать о дубликатах при импорте Import\ and\ Export=Импорт и экспорт Custom\ DOI\ URI=Пользовательский DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=Использовать пользовательский базовый URI от DOI для доступа к материалам Cited\ on\ pages=Цитаты на страницах +Please\ move\ the\ cursor\ into\ the\ document\ text.=Пожалуйста, переместите курсов в текст документа. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=Для получения информации о зрительном положении ваших цитат, мне необходимо передвинуть курсор, однако мне не удалось его захватить. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=Я не могу осуществить вставку в текущее положение курсора. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Пожалуйста, переместите курсор в положение нового цитирования. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Пожалуйста, создайте его в документе или измените в файле\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Пожалуйста, используйте последний вариант в файле оформления ниже во избежание проблем локализации. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' символа %0 является отображаемым именем для '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' символа %0 отсутствует в документе +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' абзаца %0 является отображаемым именем для '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' абзаца %0 отсутствует в документе +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Ошибка при проверке того, записывает ли Writer изменения или содержит записанные изменения. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Невозможно выполнить при включённом [Edit]/[Track Changes]/[Record]. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Изменения, выполненные JabRef, могут привести к неожиданным взаимодействиям с записанными изменениями. +Recording\ and/or\ Recorded\ changes=Изменения записываются и/или записаны +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Сперва используйте [Edit]/[Track Changes]/[Manage], чтобы их разрешить. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=Не удается найти действительный путь сертификации к запрошенной цели (%0), все равно загрузить? Download\ operation\ canceled.=Операция загрузки отменена. @@ -2340,7 +2417,6 @@ Entry\ Type=Тип записи Entry\ types=Типы записей Field\ names=Названия полей Others=Другие -Overwrite\ existing\ field\ values=Перезаписать текущие значения полей Recommended=Рекомендованный Authors\ and\ Title=Авторы и Заголовок @@ -2371,6 +2447,8 @@ Query=Запрос Question=Вопрос Select\ directory=Выберите каталог +Fulltext\ Index=Полнотекстовое индексирование +Automatically\ index\ all\ linked\ files\ for\ fulltext\ search=Автоматически индексировать все связанные файлы для полнотекстового поиска Rebuild\ fulltext\ search\ index=Перестроить полнотекствой индекс Rebuild\ fulltext\ search\ index\ for\ current\ library?=Перестроить полнотекстовой индекс для текущей библиотеки? Rebuilding\ fulltext\ search\ index...=Перестроить полнотекствой индекс... @@ -2383,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Удаленные сервисы Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Разрешить отправку PDF-файлов и цитат в онлайн сервис JabRef (Grobid) для определения метаданных. Это приводит к улучшению результатов. +Fetcher\ cannot\ be\ tested\!=Невозможно проверить сборщик данных\! +Fetcher\ unknown\!=Неизвестный сборщик данных\! Character\ by\ character=Посимвольно Embedded=Встроенный @@ -2397,7 +2477,31 @@ Could\ not\ extract\ Metadata\ from\:\ %0=Не удалось извлечь м Search\ across\ libraries\ in\ a\ new\ window=Поиск по библиотекам в новом окне Keep\ search\ string\ across\ libraries=Сохранить строку поиска по библиотекам Keep\ dialog\ always\ on\ top=Показывать диалог поверх всех окон - - - - +Search\ results\ from\ open\ libraries=Результаты поиска в открытых библиотеках + +Add\ certificate=Добавить сертификат +Serial\ number=Серийный номер +Issuer=Издатель +Valid\ from=Действителен с +Valid\ to=Действителен до +Signature\ algorithm=Алгоритм подписи +Version=Версия + +Error\ downloading=Ошибка загрузки + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Ошибка при записи метаданных. Смотрите журнал ошибок для подробностей. +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Не удалось записать метаданные, файл %1 не найден. +Success\!\ Finished\ writing\ metadata.=Успех\! Запись метаданных завершена. + +Custom\ API\ key=Пользовательский ключ API +Check\ %0\ API\ Key\ Setting=Проверьте настройку ключа API %0 + +Set=Задать +Append=Присоединить +Rename=Переименовать +New\ field\ name=Новое имя поля +Automatic\ field\ editor=Автоматический редактор поля + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Примечание\: Если в исходных записях отсутствуют ключевые слова, подходящие для конфигурации новой группы, то при подтверждении здесь они будут добавлены) +Assign=Назначить +Do\ not\ assign=Не назначать diff --git a/src/main/resources/l10n/JabRef_sv.properties b/src/main/resources/l10n/JabRef_sv.properties index 84779302d46..ae4d6c18d10 100644 --- a/src/main/resources/l10n/JabRef_sv.properties +++ b/src/main/resources/l10n/JabRef_sv.properties @@ -19,6 +19,7 @@ Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=Biblioteke %0/%1\ entries=%0/%1 poster + %0\ matches\ the\ regular\ expression\ %1=%0 matchar det reguljära uttrycket %1 %0\ matches\ the\ term\ %1=%0 matchar termen %1 @@ -138,8 +139,6 @@ Cite\ command=Citeringskommando Clear=Rensa -Clear\ fields=Rensa fält - Open\ /\ close\ entry\ editor=Öppna / stäng postredigerare Close\ dialog=Stäng dialog @@ -618,11 +617,18 @@ Remove\ selected\ entries\ from\ this\ group=Ta bort valda poster från denna gr Remove\ group=Ta bort grupp -Remove\ group\ "%0"?=Ta bort gruppen "%0"? +Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Ta bort gruppen "%0" och dess undergrupper? -Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. + + Remove\ link=Ta bort länk @@ -632,8 +638,6 @@ Remove\ string\ %0=Ta bort sträng %0 Removed\ group\ "%0".=Tog bort gruppen "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. - Removed\ string=Tog bort sträng Renamed\ string=Bytte namn på sträng @@ -685,9 +689,6 @@ Select\ entry\ type=Välj posttyp Select\ file\ from\ ZIP-archive=Välj fil från ZIP-arkiv -Set\ field=Sätt fält -Set\ fields=Sätt fält - Settings=Alternativ Shortcut=Genväg @@ -718,7 +719,6 @@ Size=Storlek Skipped\ -\ No\ PDF\ linked=Hoppade över - Ingen PDF länkad Skipped\ -\ PDF\ does\ not\ exist=Hoppade över - PDF fanns ej -Skipped\ entry.=Hoppade över post. source\ edit=ändring av källkod @@ -828,8 +828,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Rad %0 Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Rad %0\: Hittade korrupt referensnyckel %1 (kommatecken saknas). Download\ from\ URL=Ladda ned från URL Rename\ field=Byt namn på fält -Rename\ field\ to=Byt namn på fält till -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytta innehållet i ett fält till ett fält med ett annat namn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan inte använda port "%0" för fjärråtkomst; ett annat program kanske använder den. Försök med en annan port. @@ -1029,6 +1027,7 @@ Proxy\ requires\ authentication=Proxyn kräver autentisering Attention\:\ Password\ is\ stored\ in\ plain\ text\!=OBS\! Lösenordet sparas i klartext\! Clear\ connection\ settings=Rensa anslutningsinställningar + Open\ folder=Öppna mapp Export\ sort\ order=Soteringsordning vid export @@ -1370,7 +1369,6 @@ Protected\ terms\ file=Fil med skyddade termer Style\ file=Stilfil för OpenOffice/LibreOffice Open\ OpenOffice/LibreOffice\ connection=Öppna OpenOffice/LibreOffice-anslutning -You\ must\ enter\ at\ least\ one\ field\ name=Du måste ange minst ett fältnamn Non-ASCII\ encoded\ character\ found=Bokstäver som inte är ASCII-kodade hittades Toggle\ web\ search\ interface=Växla webbsökning @@ -1571,6 +1569,8 @@ Previous\ preview\ style=Föregående förhandsgranskningsstil + + diff --git a/src/main/resources/l10n/JabRef_tl.properties b/src/main/resources/l10n/JabRef_tl.properties index 0cb29b98e99..de37b3f268a 100644 --- a/src/main/resources/l10n/JabRef_tl.properties +++ b/src/main/resources/l10n/JabRef_tl.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 nag tugma sa regular na ekspresyon %1 %0\ matches\ the\ term\ %1=%0 tumugma sa termino %1 @@ -126,8 +127,6 @@ Cite\ command=Utos na nabanggit Clear=Malinaw -Clear\ fields=Malinaw na patlang - Close\ dialog=Isara ang dialog @@ -600,11 +599,18 @@ Remove\ selected\ entries\ from\ this\ group=Tanggalin ang napiling entries mula Remove\ group=Tanggaling ang grupo -Remove\ group\ "%0"?=Tanggalin ang grupo "%0"? +Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Tanggalin ang grupo "%0" at ang mababang grupo? -Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo +Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. + + Remove\ link=Tanggalin ang link @@ -613,8 +619,6 @@ Remove\ old\ entry=Tanggalin ang lumang entry Removed\ group\ "%0".=Tanggalin ang grupo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. - Removed\ string=Na tanggal ang string Renamed\ string=Palitan ulit ng pangalan ang string @@ -671,9 +675,6 @@ Select\ file\ from\ ZIP-archive=Piliin ang file mula sa na arkiba na ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Piliin ang nodes na puno para makita at matanggap o matanggihan ang pagbabago -Set\ field=Ihanda ang patlang -Set\ fields=Ihanda ang patlang - Settings=Mga settings Shortcut=Shortcut @@ -706,7 +707,6 @@ Size=Laki Skipped\ -\ No\ PDF\ linked=Nilaktawan - Walang PDF ang na linked Skipped\ -\ PDF\ does\ not\ exist=Nilaktawan - PDF ay hindi umiiral -Skipped\ entry.=Nilaktawan ang entry. source\ edit=i-edit ang pinagmulan @@ -820,10 +820,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=I-download mula sa URL Rename\ field=Palitan ang pangalan ng patlang -Append\ field=Ilagay ang field -Append\ to\ fields=Ilagay sa mga patlang -Rename\ field\ to=Palitan ang pangalan ng patlang sa -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Ilipat ang mga nilalaman ng isang patlang sa isang patlang na may ibang pangalan Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Hindi magagamit ang port %0 para sa remote na operasyon; maaaring gamitin ng isa pang application. Subukan ang pagtukoy ng isa pang port. @@ -1033,6 +1029,7 @@ Proxy\ requires\ authentication=Kinakailangan ng proxy ang pagpapatunay Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Pansin\: Ang password ay naka-imbak sa plain text\! Clear\ connection\ settings=I-clear ang mga setting ng koneksyon + Export\ sort\ order=Mag-export ng uri ng order Show\ extra\ columns=Magpakita ng mga dagdag na haligi @@ -1206,7 +1203,6 @@ Protected\ terms\ file=Protektadong mga tuntunin na file Style\ file=Estilo ng file Open\ OpenOffice/LibreOffice\ connection=Buksan ang OpenOffice/LibreOffice na koneksyon -You\ must\ enter\ at\ least\ one\ field\ name=Dapat kang magpasok ng hindi bababa sa isang pangalan ng field Non-ASCII\ encoded\ character\ found=Natagpuan ang naka-encode na karakter na hindi-ASCII Toggle\ web\ search\ interface=I-toggle ang interface ng paghahanap sa web @@ -1276,7 +1272,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Tinanggal ang l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Hindi makuha ang data mula sa entry '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Ang entry mula sa %0 ay hindi ma-parse. Invalid\ identifier\:\ '%0'.=Di-wastong identifier\: '%0'. -This\ paper\ has\ been\ withdrawn.=Ang papel na ito ay na-withdraw. Aux\ file=Aux file Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo na naglalaman ng mga entry na binanggit sa isang ibinigay na file ng TeX @@ -1364,6 +1359,8 @@ Default\ pattern=Default na pattern + + diff --git a/src/main/resources/l10n/JabRef_tr.properties b/src/main/resources/l10n/JabRef_tr.properties index 80d892c18af..fc09ebe0ce1 100644 --- a/src/main/resources/l10n/JabRef_tr.properties +++ b/src/main/resources/l10n/JabRef_tr.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 girdi -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 dışa aktarım başarılı. Kaydedilen dosyayı içeren klasör açılsın mı? + %0\ matches\ the\ regular\ expression\ %1=%0 şu Düzenli İfadeyle eşleşiyor %1 @@ -151,8 +151,6 @@ Cite\ command=Alıntı komutu Clear=Sil -Clear\ fields=Alanları sil - Open\ /\ close\ entry\ editor=Girdi düzenleyicisini aç/kapat Close\ dialog=Dialoğu kapat @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Seçili girdileri bu gruptan çıka Remove\ group=Grubu sil -Remove\ group\ "%0"?="%0" grubu silinsin mi? +Remove\ group\ and\ subgroups=Grubu ve altgrupları sil + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?="%0" grubu ve altgrupları silinsin mi? -Remove\ group\ and\ subgroups=Grubu ve altgrupları sil +Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. + + Remove\ link=Linki sil @@ -717,8 +722,6 @@ Remove\ string\ %0=%0 Dizgesini sil Removed\ group\ "%0".="%0" grubu silindi. -Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. - Removed\ string=Dizge silindi Renamed\ string=Dizge yeniden adlandırıldı @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=ZIP arşivinden dosyayı seçiniz Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Değişiklikleri görmek ve kabul ya da reddetmek için ağaç düğümlerini seçiniz -Set\ field=Alanı ata -Set\ fields=Alanları ata - Settings=Ayarlar Shortcut=Kısayol @@ -825,7 +825,6 @@ Size=Boyut Skipped\ -\ No\ PDF\ linked=Atlandı - PDF eklenmedi Skipped\ -\ PDF\ does\ not\ exist=Atlandı - PDF mevcut değil -Skipped\ entry.=Girdi atlandı. Sort\ subgroups=Alt grupları sırala @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Satır %0\: B No\ full\ text\ document\ found=Tam metin belge bulunamadı Download\ from\ URL=URL'den indir Rename\ field=Alanın adını değiştir -Append\ field=Alan ekle -Append\ to\ fields=Alanlara ekle -Rename\ field\ to=Alan adını şuna değiştir -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Alan içeriğini başka isimli bir alanın içine taşı Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Uzak operasyon için bağlantı noktası %0 kullanılamıyor; bir başka program kullanıyor olabilir. Başka bir bağlantı noktası deneyin. @@ -1243,6 +1238,15 @@ Use\ custom\ proxy\ configuration=Özelleştirilmiş vekil konfigürasyonu kulla Proxy\ requires\ authentication=Vekil sunucu kimlik denetleme gerektiriyor Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Dikkat\: Parola salt metin olarak kaydedildi\! Clear\ connection\ settings=Bağlantı ayarlarını sil +Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et +Check\ connection=Bağlantıyı kontrol et +Connection\ failed\!=Bağlantı başarısız\! +Connection\ successful\!=Bağlantı başarılı\! + +SSL\ Configuration=SSL Yapılandırması +SSL\ certificate\ file=SSL sertifika dosyası +Duplicate\ Certificates=Çift nüsha Sertifikalar +You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Open\ folder=Klasörü aç Export\ sort\ order=Sıralama kriterlerini dışa aktar @@ -1703,7 +1707,6 @@ Protected\ terms\ file=Korunmuş terimler dosyası Style\ file=Stil dosyası Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice bağlantısı aç -You\ must\ enter\ at\ least\ one\ field\ name=En az bir alan adı girmelisiniz Non-ASCII\ encoded\ character\ found=ASCII koduyla kodlanmamış karakter bulundu Non-UTF-8\ encoded\ field\ found=UTF-8 ile kodlanmamış alan bulundu Toggle\ web\ search\ interface=Ağ arama arayüzünü değiştir @@ -1841,7 +1844,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Alan içeriğin Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' dan girdi verileri alınamadı. Entry\ from\ %0\ could\ not\ be\ parsed.=%0'dan girdi çözümlenemedi. Invalid\ identifier\:\ '%0'.=Geçersiz tanımlayıcı\: '%0 '. -This\ paper\ has\ been\ withdrawn.=Bu yayın geri çekildi. empty\ citation\ key=boş atıf anahtarı Aux\ file=Aux dosya Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Belirli bir TeX dosyasında alıntılanmış girdileri içeren grup @@ -2324,10 +2326,6 @@ Removes\ digits.=Rakamları siler. Presets=Ön ayarlar -Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et -Check\ connection=Bağlantıyı kontrol et -Connection\ failed\!=Bağlantı başarısız\! -Connection\ successful\!=Bağlantı başarılı\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Aşağıdaki alandaki anahtar sözcüklerden gruplar oluştur Generate\ groups\ for\ author\ last\ names=Yazar soyadları için gruplar oluştur Regular\ expression=Düzenli ifade @@ -2400,7 +2398,6 @@ Entry\ Type=Girdi Türü Entry\ types=Girdi türleri Field\ names=Alan adları Others=Diğerleri -Overwrite\ existing\ field\ values=Mevcut alan değerlerinin üzerine yaz Recommended=Önerilen Authors\ and\ Title=Yazarlar ve Başlık @@ -2445,6 +2442,8 @@ Grobid\ URL=Grobid URL Remote\ services=Uzak servisler Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Metaverisini belirlemek için PDF dosyalarını ve ham atıf dizgelerini bir JabRef çevrim içi servisine (Grobid) göndermeye izin ver. Bu daha iyi sonuçlara yol açar. +Fetcher\ cannot\ be\ tested\!=Getirici test edilemiyor\! +Fetcher\ unknown\!=Getirici bilinmiyor\! Character\ by\ character=Karakter karakter Embedded=Gömülü @@ -2468,10 +2467,6 @@ Valid\ from=Geçerlilik tarihi Valid\ to=Son kullanma tarihi Signature\ algorithm=İmza algoritması Version=Sürüm -SSL\ Configuration=SSL Yapılandırması -SSL\ certificate\ file=SSL sertifika dosyası -Duplicate\ Certificates=Çift nüsha Sertifikalar -You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Error\ downloading=İndirmede hata @@ -2479,3 +2474,7 @@ Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Metaverisi Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metaverisi yazma başarısız, %1 dosyası bulunamadı. Success\!\ Finished\ writing\ metadata.=Başarılı\! Metaverisi yazma bitirildi. +Custom\ API\ key=Özel API anahtarı +Check\ %0\ API\ Key\ Setting=%0 API Anahtar Ayarlarını Kontrol Edin + + diff --git a/src/main/resources/l10n/JabRef_vi.properties b/src/main/resources/l10n/JabRef_vi.properties index 26aaa90bf04..d82df8e5422 100644 --- a/src/main/resources/l10n/JabRef_vi.properties +++ b/src/main/resources/l10n/JabRef_vi.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 khớp biểu thức chính tắc %1 %0\ matches\ the\ term\ %1=%0 khớp thuật ngữ %1 @@ -133,8 +134,6 @@ Cite\ command=Lệnh trích dẫn Clear=Xóa -Clear\ fields=Xóa các dữ liệu - Close\ dialog=Đóng hộp thoại @@ -612,11 +611,18 @@ Remove\ entry\ from\ import=Loại bỏ mục khỏi lệnh nhập Remove\ group=Loại bỏ nhóm -Remove\ group\ "%0"?=Loại bỏ nhóm "%0"? +Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Loại bỏ nhóm "%0" và các nhóm con của nó? -Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con +Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. + + Remove\ link=Loại bỏ liên kết @@ -625,8 +631,6 @@ Remove\ old\ entry=Loại bỏ mục cũ Removed\ group\ "%0".=Đã loại bỏ nhóm "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. - Removed\ string=Đã loại bỏ chuỗi Renamed\ string=Chuỗi được đặt tên lại @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Chọn tập tin từ tập tin ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Chọn các nốt trên sơ đồ hình cây để xem và chấp nhận hoặc từ chối thay đổi -Set\ field=Thiết lập dữ liệu -Set\ fields=Thiết lập các dữ liệu - Settings=Các thiết lập Shortcut=Phím tắt @@ -712,7 +713,6 @@ Size=Kích thước Skipped\ -\ No\ PDF\ linked=Bị bỏ qua - Không có tập tin PDF được liên kết Skipped\ -\ PDF\ does\ not\ exist=Bỏ qua - tập tin PDF không tồn tại -Skipped\ entry.=Mục bị bỏ qua. source\ edit=chỉnh sửa nguồn @@ -826,8 +826,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Kéo từ URL Rename\ field=Đổi tên dữ liệu -Rename\ field\ to=Đổi tên dữ liệu thành -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Di chuyển nội dung của một dữ liệu sang một dữ liệu có tên khác Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Không thể dùng cổng %0 cho lệnh chạy từ xa; một ứng dụng khác có thể đang dùng nó. Hãy thử chỉ định một cổng khác. @@ -900,6 +898,7 @@ Finished\ automatically\ setting\ external\ links.=Thiết lập tự động c + Open\ folder=Mở thư mục @@ -1135,6 +1134,8 @@ Default\ pattern=Kiểu mặc định + + diff --git a/src/main/resources/l10n/JabRef_zh_CN.properties b/src/main/resources/l10n/JabRef_zh_CN.properties index 0577de6a7b9..baca5d6da46 100644 --- a/src/main/resources/l10n/JabRef_zh_CN.properties +++ b/src/main/resources/l10n/JabRef_zh_CN.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 条目 +Export\ operation\ finished\ successfully.=导出成功。 + +Reveal\ in\ File\ Explorer=在文件夹中显示 %0\ matches\ the\ regular\ expression\ %1=%0 匹配正则表达式 %1 @@ -28,7 +31,7 @@ Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (DEFAULT\ abbreviation)= Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (MEDLINE\ abbreviation)=缩写选中记录的期刊名 (MEDLINE 格式缩写) Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (SHORTEST\ UNIQUE\ abbreviation)=缩写选中记录的期刊名 (最短唯一 (SHORTEST UNIQUE) 格式缩写) -Abbreviate\ names=缩写名 +Abbreviate\ names=缩写名(J. Smith) Abbreviated\ %0\ journal\ names.=缩写的 %0 期刊名称。 Abbreviation=缩写 @@ -150,8 +153,6 @@ Cite\ command=引用命令 Clear=清除 -Clear\ fields=清除字段内容 - Open\ /\ close\ entry\ editor=打开/关闭条目编辑器 Close\ dialog=关闭对话框 @@ -201,7 +202,7 @@ Could\ not\ run\ the\ 'vim'\ program.=无法运行 'vim' 程序。 Could\ not\ save\ file.=无法保存文件 Character\ encoding\ '%0'\ is\ not\ supported.=,不支持编码 '%0'。 -Create\ custom\ fields\ for\ each\ BibTeX\ entry=为每个 BibTeX 条目创建自定义字段 +Create\ custom\ fields\ for\ each\ BibTeX\ entry=Create custom fields for each BibTeX entry crossreferenced\ entries\ included=包含交叉引用的记录 @@ -265,7 +266,7 @@ Display\ help\ on\ command\ line\ options=用命令行选项显示帮助 Display\ only\ entries\ belonging\ to\ all\ selected\ groups=仅显示所选分组的交集条目 Display\ version=显示版本 -Do\ not\ abbreviate\ names=不要缩写姓名 +Do\ not\ abbreviate\ names=不缩写名(John Smith) Do\ not\ import\ entry=不导入记录 @@ -376,9 +377,9 @@ First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=首先选 Fit\ table\ horizontally\ on\ screen=列表宽度填满屏幕宽度 Float=浮动 (结果上浮到最前) -Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式\: 选项卡\: 字段; 字段; ...(例如 General\:url;pdf;note...) +Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式:选项卡\:字段;字段;...(例如:General\:url;pdf;note...) -Format\ of\ author\ and\ editor\ names=作者和编者的姓名格式 +Format\ of\ author\ and\ editor\ names=author和editor的姓名格式 Format\ string=格式化简写字串 Format\ used=使用的格式 @@ -492,7 +493,7 @@ Key\ bindings=热键 Key\ bindings\ changed=热键已更改 -Key\ pattern=键值表达式 +Key\ pattern=Key pattern keys\ in\ library=文献库中的键值 @@ -541,7 +542,9 @@ Moved\ group\ "%0".=移动了分组 "%0"。 Mr.\ DLib\ Privacy\ settings=Mr. DLib 隐私设置 +No\ database\ is\ open=No database is open +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=We need a database to export from. Open one. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=没有收到 Mr. DLib 对于本条目的建议。 @@ -549,9 +552,9 @@ Error\ while\ fetching\ recommendations\ from\ Mr.DLib.=在接收 Mr. DLib 的 Name=名称 -Name\ formatter=姓名格式化器 +Name\ formatter=姓名格式化 -Natbib\ style=Natbib 格式 +Natbib\ style=Natbib样式 nested\ AUX\ files=nested AUX 文件 @@ -700,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=从这个组里移除选定记录 Remove\ group=移除分组 -Remove\ group\ "%0"?=移除分组 "%0"? +Remove\ group\ and\ subgroups=移除分组和子分组 + +Remove\ groups\ and\ subgroups=移除分组和子分组 + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=移除所有选定的分组但保留其子分组? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=移除分组 "%0" 但保留其子分组? + +Remove\ groups=移除分组 + +Removed\ all\ selected\ groups.=移除选中的分组 Remove\ group\ "%0"\ and\ its\ subgroups?=移除分组 "%0" 和它的子分组? -Remove\ group\ and\ subgroups=移除分组和子分组 +Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=移除所有选定的分组及其子分组? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=移除所有选定的分组及其子分组 Remove\ link=移除链接 @@ -714,8 +731,6 @@ Remove\ string\ %0=移除字符串 %0 Removed\ group\ "%0".=已移除分组 "%0"。 -Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 - Removed\ string=已移除简写字串 Renamed\ string=重命名简写字串 @@ -753,7 +768,7 @@ Review=评论 Review\ changes=复查修改 Review\ Field\ Migration=查看字段迁移 -Loading=加载中 +Loading=启动 Save=保存 Save\ all\ finished.=完成保存全部。 @@ -764,10 +779,10 @@ Save\ before\ closing=关闭前保存 Save\ library=保存 Save\ library\ as...=另存为 -Saving=保存中 +Saving=保存 Saving\ all\ libraries...=正在保存所有数据库... Saving\ library=正在保存文献库 -Library\ saved=Library saved +Library\ saved=文献库已保存 Saved\ selected\ to\ '%0'.=保存选中到 '%0'. Search=查找 @@ -787,24 +802,21 @@ Select\ file\ from\ ZIP-archive=从 ZIP-压缩包中选择文件 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=选择树节点查看和接受/拒绝修改 -Set\ field=设置字段内容 -Set\ fields=设置字段内容 - Settings=设置 Shortcut=快捷键 Show/edit\ %0\ source=显示/编辑 %0 源代码 -Show\ 'Firstname\ Lastname'=显示 '名 (Firstname) 姓 (Lastname)' +Show\ 'Firstname\ Lastname'=名在前,姓在后(John Smith) -Show\ 'Lastname,\ Firstname'=显示 '姓 (Lastname), 名 (Firstname)' +Show\ 'Lastname,\ Firstname'=姓在前,名在后(Smith, John) Show\ BibTeX\ source\ by\ default=默认显示 BibTeX 源代码 Show\ confirmation\ dialog\ when\ deleting\ entries=删除多条记录时发出警告 -Show\ last\ names\ only=只显示“姓 (Lastname)” +Show\ last\ names\ only=仅显示姓(Smith) Show\ names\ unchanged=显示原始姓名字串 @@ -822,12 +834,17 @@ Size=大小 Skipped\ -\ No\ PDF\ linked=跳过 - 没有 PDF 链接 Skipped\ -\ PDF\ does\ not\ exist=跳过 - PDF 不存在 -Skipped\ entry.=已跳过记录 +JabRef\ skipped\ the\ entry.=JabRef 跳过了条目。 +Import\ error=导入错误 +Open\ library\ error=打开文献库出错 +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=请检查您的库文件是否存在语法错误。 +SourceTab\ error=SourceTab错误 +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=通过条目编辑器的 "{}bibtex source" 标签中输入时出错。 Sort\ subgroups=排序子组 source\ edit=源代码编辑 -Special\ name\ formatters=特殊的姓名格式化器 +Special\ name\ formatters=姓名格式化 Statically\ group\ entries\ by\ manual\ assignment=手动创建静态分组 @@ -956,7 +973,7 @@ Refuse\ to\ save\ the\ library\ before\ external\ changes\ have\ been\ reviewed. Library\ protection=文献库保护 Unable\ to\ save\ library=无法保存文献库 -Citation\ key\ generator=Citationkey生成器 +Citation\ key\ generator=Citationkey Unable\ to\ open\ link.=无法打开链接。 MIME\ type=MIME 类型 @@ -969,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=第 %0 行\: No\ full\ text\ document\ found=未发现完整的文档 Download\ from\ URL=从 URL 下载 Rename\ field=重命名字段 -Append\ field=添加字段 -Append\ to\ fields=追加到字段 -Rename\ field\ to=重命名字段为 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=将某个字段的内容移动到另一个字段中 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=无法使用端口 %0 进行远程操作;该端口可能被其它应用程序占用,请使用其它端口。 @@ -1042,6 +1055,7 @@ Cite\ special=引用特殊 Extra\ information\ (e.g.\ page\ number)=额外信息(例如:页码) Manage\ citations=管理文献引用 Problem\ modifying\ citation=修改文献引用存在问题 +Problem\ collecting\ citations=Problem collecting citations Citation=文献引用 Connecting...=连接中... Select\ style=选择引用样式 @@ -1056,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=同步 OpenOffice/LibreOffice 参考 Select\ which\ open\ Writer\ document\ to\ work\ on=选择使用哪个打开的 Writer 文档 Connected\ to\ document=连接到文档 +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Could not connect to any Writer document. Please make sure a document is open before using the 'Select Writer document' button to connect to it. +No\ Writer\ documents\ found=No Writer documents found Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=插入一条没有内容的引用(这条记录将会出现在引用列表中) Cite\ selected\ entries\ with\ extra\ information=引用包含额外信息的选中记录 @@ -1064,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=保证参考文献是最新的 Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=您的OpenOffice/LibreOffice文档引用了 citationkey "%0",但未在当前库中找到。 +This\ operation\ requires\ a\ bibliography\ database.=This operation requires a bibliography database. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Your OpenOffice/LibreOffice document references at least %0 citation keys which could not be found in your current library. Some of these are %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=Your OpenOffice/LibreOffice document references no citation keys which could also be found in your current library. Unable\ to\ synchronize\ bibliography=无法同步参考文献 Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=合并仅仅由空格分隔的两段引文 @@ -1073,7 +1092,12 @@ Autodetection\ failed=自动检测失败 Please\ wait...=请稍候... Connection\ lost=连接丢失 +Could\ not\ update\ bibliography=Could not update bibliography +Not\ connected\ to\ document=Not connected to document +Problem\ combining\ cite\ markers=Problem combining cite markers +Unable\ to\ reload\ style\ file=Unable to reload style file +Problem\ during\ separating\ cite\ markers=Problem during separating cite markers Automatically\ sync\ bibliography\ when\ inserting\ citations=当插入文献引用时自动同步参考文献 Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=在当前标签查找 BibTeX 记录 @@ -1176,6 +1200,7 @@ Toggle\ relevance=标记为相关 Toggle\ quality\ assured=标记为质量已确认 Toggle\ print\ status=标记打印状态 Update\ keywords=更新关键词 +Problem\ connecting=Problem connecting Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=与OpenOffice/LibreOffice的连接已丢失。请确保OpenOffice/LibreOffice正在运行,并尝试重新连接。 JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef 将会为每个条目发送至少一个请求至发布者。 @@ -1228,6 +1253,16 @@ Use\ custom\ proxy\ configuration=使用自定义 HTTP 代理 Proxy\ requires\ authentication=代理服务器需要密码 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意\: 密码以明文形式保存\! Clear\ connection\ settings=清除连接设置 +Check\ Proxy\ Setting=检查代理设置 +Check\ connection=检查连接 +Connection\ failed\!=连接失败! +Connection\ successful\!=连接成功! + +SSL\ Configuration=SSL配置 +SSL\ configuration\ changed=SSL配置已更改 +SSL\ certificate\ file=SSL证书文件 +Duplicate\ Certificates=复制证书 +You\ already\ added\ this\ certificate=您已经添加了此证书 Open\ folder=打开文件夹 Export\ sort\ order=导出顺序 @@ -1264,6 +1299,7 @@ Please\ open\ %0\ manually.=请手动打开 %0。 The\ link\ has\ been\ copied\ to\ the\ clipboard.=链接已复制到剪贴板 Open\ %0\ file=打开文件 %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=无法自动检测终端。请在首选项中自定义一个终端。 Cannot\ delete\ file=无法删除文件 File\ permission\ error=文件权限错误 @@ -1365,7 +1401,7 @@ Automatically\ setting\ file\ links=自动设置文件链接 Regenerating\ citation\ keys\ according\ to\ metadata=根据元数据重新生成 citationkey Regenerate\ all\ keys\ for\ the\ entries\ in\ a\ BibTeX\ file=重新生成 BibTeX 文件中所有记录的键值 Show\ debug\ level\ messages=显示调试级别消息 -Default\ library\ mode=Default library mode +Default\ library\ mode=默认文献库模式 Show\ only\ preferences\ deviating\ from\ their\ default\ value=只显示与默认值不同的首选项 default=默认 key=键值 @@ -1570,10 +1606,21 @@ Custom=自定义 Export\ cited=导出已被引用的 Unable\ to\ generate\ new\ library=无法生成新库 +The\ cursor\ is\ in\ a\ protected\ area.=The cursor is in a protected area. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=No bibliography database is open for citation. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=No database is open for updating citation markers after citing. +No\ bibliography\ entries\ are\ selected\ for\ citation.=No bibliography entries are selected for citation. +No\ bibliography\ style\ is\ selected\ for\ citation.=No bibliography style is selected for citation. +No\ database=No database +No\ entries\ selected\ for\ citation=No entries selected for citation +Open\ one\ before\ citing.=Open one before citing. +Select\ one\ before\ citing.=Select one before citing. +Select\ some\ before\ citing.=Select some before citing. Found\ identical\ ranges=Found identical ranges Found\ overlapping\ ranges=Found overlapping ranges @@ -1621,7 +1668,7 @@ Remove\ list=删除列表 Add\ abbreviation=添加缩写 Full\ journal\ name=期刊全称 Abbreviation\ name=缩写名 -Shortest\ unique\ abbreviation=Shortest unique abbreviation +Shortest\ unique\ abbreviation=最简缩写 No\ abbreviation\ files\ loaded=缩写文件未加载 @@ -1640,6 +1687,8 @@ Issue\ report\ successful=问题报告成功 Your\ issue\ was\ reported\ in\ your\ browser.=您的问题已在浏览器中报告。 The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=日志和异常信息已复制到剪贴板中。 Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=请将此信息粘贴至问题描述中 (用 Ctrl + V)。 +Last\ notification=最新通知 +Check\ the\ event\ log\ to\ see\ all\ notifications=在事件日志中查看所有通知 Host=主机 Port=端口 @@ -1677,7 +1726,6 @@ Protected\ terms\ file=受保护的术语文件 Style\ file=样式文件 Open\ OpenOffice/LibreOffice\ connection=打开 OpenOffice/LibreOffice 连接 -You\ must\ enter\ at\ least\ one\ field\ name=您至少需要输入一个字段名 Non-ASCII\ encoded\ character\ found=发现Non-ASCII编码字符 Non-UTF-8\ encoded\ field\ found=未找到 UTF-8 编码的字段 Toggle\ web\ search\ interface=切换网页搜索面板 @@ -1758,7 +1806,7 @@ Unhandled\ exception\ occurred.=发生了无法处理的异常。 strings\ included=包含的字符串 Escape\ underscores=转义下划线 -Color=色彩 +Color=颜色 Please\ also\ add\ all\ steps\ to\ reproduce\ this\ issue,\ if\ possible.=如果可能的话,也请添加所有步骤以重现这个问题。 Fit\ width=适应宽度 Fit\ a\ single\ page=适应单页 @@ -1815,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=移除字段内 Could\ not\ retrieve\ entry\ data\ from\ '%0'.=无法从 "%0" 中检索条目数据。 Entry\ from\ %0\ could\ not\ be\ parsed.=无法解析 %0 中的条目。 Invalid\ identifier\:\ '%0'.=无效的标识符:'%0'。 -This\ paper\ has\ been\ withdrawn.=这篇论文已被撤回。 empty\ citation\ key=Citationkey为空 Aux\ file=Aux 文件 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=包含给定TeX文件中引用的条目组 @@ -1883,7 +1930,7 @@ View\ change\ log=查看变更记录 View\ event\ log=查看事件日志 Website=网站 -Override\ default\ font\ settings=跳过默认字体设置 +Override\ default\ font\ settings=覆盖默认的字体设置 Clear\ search=清除搜索 Click\ help\ to\ learn\ about\ the\ migration\ of\ pre-3.6\ databases.=点击帮助了解 pre-3.6 数据库的迁移情况。 @@ -1993,47 +2040,47 @@ Import\ entries\ from\ LaTeX\ files=Import entries from LaTeX files Import\ new\ entries=Import new entries Group\ color=分组颜色 -Columns=Columns +Columns=列 File\ type=File type IEEE=IEEE Internal=Internal Special=Special Remove\ column=Remove column -Add\ custom\ column=Add custom column -Update\ to\ current\ column\ order=Update to current column order -Sort\ column\ one\ step\ upwards=Sort column one step upwards -Sort\ column\ one\ step\ downwards=Sort column one step downwards +Add\ custom\ column=添加自定义列 +Update\ to\ current\ column\ order=更新到当前的列顺序 +Sort\ column\ one\ step\ upwards=上移一位 +Sort\ column\ one\ step\ downwards=下移一位 List\ must\ not\ be\ empty.=List must not be empty. -Order=Order +Order=排序 -Add\ field\ to\ filter\ list=Add field to filter list +Add\ field\ to\ filter\ list=添加字段到过滤列表 Add\ formatter\ to\ list=Add formatter to list Filter\ List=Filter List Open\ files...=打开文件... Affected\ fields\:=补全字段 -Show\ preview\ as\ a\ tab\ in\ entry\ editor=Show preview as a tab in entry editor -Font=Font -Visual\ theme=Visual theme +Show\ preview\ as\ a\ tab\ in\ entry\ editor=在条目编辑器中以标签形式显示预览 +Font=字体 +Visual\ theme=主题风格 Light\ theme=明亮主题 Dark\ theme=暗色主题 Custom\ theme=自定义 Overwrite\ existing\ keys=覆盖该字段的内容 -Key\ patterns=不同类型条目的键值风格 +Key\ patterns=Citationkey样式 Font\ settings=Font settings Please\ specify\ a\ css\ theme\ file.=请指定一个 css 主题文件。 You\ must\ enter\ an\ integer\ value\ higher\ than\ 8.=You must enter an integer value higher than 8. Letters\ after\ duplicate\ generated\ keys=为重复的Citationkey添加字母后缀 -Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 A (a, b, ...) -Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 B (b, c, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 a (a, b, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 b (b, c, ...) Always\ add\ letter\ (a,\ b,\ ...)\ to\ generated\ keys=在生成键值时总是添加字母 (a, b, ...) -Default\ pattern=默认模式 +Default\ pattern=Default pattern Reset\ %s\ to\ default\ value=Reset %s to default value Library\ mode=Library mode Reset\ to\ recommended=Reset to recommended Remove\ all=Remove all -Reset\ All=Reset All +Reset\ All=全部重置 Linked\ identifiers=Linked identifiers insert\ entries=insert entries @@ -2056,13 +2103,14 @@ Explicit\ selection=手动选择 Searching\ for\ a\ keyword=指定字段的关键词 Free\ search\ expression=检索语句 Specified\ keywords=字段内容 -Cited\ entries=Cited entries +Cited\ entries=根据AUX文件 Search\ term\ is\ empty.=Search term is empty. Invalid\ regular\ expression.=无效的正则表达式 Please\ provide\ a\ valid\ aux\ file.=请提供一个有效的AUX文件 Keyword\ delimiter=Keyword delimiter Hierarchical\ keyword\ delimiter=Hierarchical keyword delimiter Escape\ ampersands=转义逻辑"与"符号 +Escape\ dollar\ sign=Escape dollar sign Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Hint\:\n\nTo search all fields for Smith, enter\:\nsmith\n\nTo search the field author for Smith and the field title for electrical, enter\:\nauthor\=Smith and title\=electrical @@ -2272,7 +2320,7 @@ Reveal\ in\ file\ explorer=Reveal in file explorer Autolink\ files=自动链接文件 -Custom\ editor\ tabs=条目编辑器的标签 +Custom\ editor\ tabs=条目编辑器 Custom\ export\ formats=自定义导出格式 Custom\ import\ formats=自定义导入格式 @@ -2283,7 +2331,7 @@ Customized\ preview\ style=自定义预览样式 Next\ preview\ style=下一个预览样式 Previous\ preview\ style=上一个预览样式 -(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=( Note\: Press return to commit changes in the table\! ) +(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=(注意:按回车键以确认更改) Reset=Reset New\ inproceedings=新的inproceedings Reset\ entry\ types\ and\ fields\ to\ defaults=Reset entry types and fields to defaults @@ -2297,10 +2345,6 @@ Removes\ digits.=Removes digits. Presets=默认 -Check\ Proxy\ Setting=检查代理设置 -Check\ connection=检查连接 -Connection\ failed\!=连接失败! -Connection\ successful\!=连接成功! Generate\ groups\ from\ keywords\ in\ the\ following\ field=根据以下字段的关键字生成组 Generate\ groups\ for\ author\ last\ names=根据作者的姓氏生成组 Regular\ expression=正则表达式 @@ -2328,19 +2372,34 @@ Custom\ DOI\ URI=自定义 DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=使用基于DOI的URI访问文章 Cited\ on\ pages=Cited on pages +Please\ move\ the\ cursor\ into\ the\ document\ text.=Please move the cursor into the document text. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=To get the visual positions of your citations I need to move the cursor around, but could not get it. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=I cannot insert to the cursors current location. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Please move the cursor to the location for the new citation. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Please create it in the document or change in the file\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Please use the latter in the style file below to avoid localization problems. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 character style '%1' is a display name for '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 character style '%1' is missing from the document +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 paragraph style '%1' is a display name for '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 paragraph style '%1' is missing from the document +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Error while checking if Writer is recording changes or has recorded changes. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Cannot work with [Edit]/[Track Changes]/[Record] turned on. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Changes by JabRef could result in unexpected interactions with recorded changes. +Recording\ and/or\ Recorded\ changes=Recording and/or Recorded changes +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Use [Edit]/[Track Changes]/[Manage] to resolve them first. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=无法找到请求目标(%0)的有效证书路径,仍然下载? Download\ operation\ canceled.=下载操作已取消。 @@ -2358,7 +2417,6 @@ Entry\ Type=条目类型 Entry\ types=条目类型 Field\ names=字段名 Others=其他 -Overwrite\ existing\ field\ values=覆盖现有字段内容 Recommended=推荐 Authors\ and\ Title=作者和标题 @@ -2401,8 +2459,10 @@ Found\ matches\ in\ Annotations\:=在注释中找到匹配项: Grobid\ URL=Grobid URL Remote\ services=远程服务 -Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Allow sending PDF files and raw citation strings to a JabRef online service (Grobid) to determine Metadata. This produces better results. +Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=允许将 PDF 文件和原始引用发送到JabRef在线服务(Grobid)以获取元数据 +Fetcher\ cannot\ be\ tested\!=Fetcher cannot be tested\! +Fetcher\ unknown\!=Fetcher unknown\! Character\ by\ character=Character by character Embedded=内嵌 @@ -2419,6 +2479,29 @@ Keep\ search\ string\ across\ libraries=在所有库中保留搜索语句 Keep\ dialog\ always\ on\ top=保持窗口位于最前面 Search\ results\ from\ open\ libraries=Search results from open libraries +Add\ certificate=添加证书 +Serial\ number=序号 +Issuer=签发者 +Valid\ from=起始有效期 +Valid\ to=截至有效期 +Signature\ algorithm=签名算法 +Version=版本 + +Error\ downloading=下载出错 + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=写入元数据时出错。详细信息请参阅错误日志。 +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Failed to write metadata, file %1 not found. +Success\!\ Finished\ writing\ metadata.=Success\! Finished writing metadata. +Custom\ API\ key=自定义API +Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting +Set=设定 +Append=附加 +Rename=重命名 +New\ field\ name=新的字段名 +Automatic\ field\ editor=自动化条目编辑 +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note\: If original entries lack keywords to qualify for the new group configuration, confirming here will add them) +Assign=Assign +Do\ not\ assign=Do not assign diff --git a/src/main/resources/l10n/JabRef_zh_TW.properties b/src/main/resources/l10n/JabRef_zh_TW.properties index 588f8dd4d77..288532048c0 100644 --- a/src/main/resources/l10n/JabRef_zh_TW.properties +++ b/src/main/resources/l10n/JabRef_zh_TW.properties @@ -15,6 +15,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 條目 + %0\ matches\ the\ regular\ expression\ %1=%0 符合正規表達式 %1 %0\ matches\ the\ term\ %1=%0 符合字詞 %1 @@ -142,8 +143,6 @@ Cite\ command=引用指令 Clear=清除 -Clear\ fields=清除欄位 - Close\ dialog=關閉對話視窗 @@ -579,6 +578,13 @@ Remove\ group=移除群組 + + + + + + + Remove\ link=移除連結 Remove\ old\ entry=移除舊有條目 @@ -588,7 +594,6 @@ Remove\ string\ %0=移除字串 %0 - Replace=取代 Replace\ With\:=取代為: Find\:=尋找: @@ -627,7 +632,6 @@ Select\ all=全選 - Settings=設定 Shortcut=快速鍵 @@ -728,8 +732,6 @@ Unable\ to\ save\ library=無法儲存文獻倉庫 Rename\ field=重新命名欄位 -Append\ field=添加欄位 -Append\ to\ fields=添加到欄位 Autosave\ local\ libraries=自動儲存本機文獻倉庫 @@ -806,6 +808,11 @@ Proxy\ configuration=代理伺服器設定 Use\ custom\ proxy\ configuration=使用自訂代理伺服器 Proxy\ requires\ authentication=代理伺服器需要認證 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意:密碼將以明文形式存放! +Check\ Proxy\ Setting=檢查代理設定 +Check\ connection=檢查連接 +Connection\ failed\!=連線失敗! +Connection\ successful\!=連線成功! + Export\ sort\ order=匯出順序 @@ -984,7 +991,7 @@ Set\ rank\ to\ five=設定評分為 5 級 Order=順序 -Affected\ fields\:=影響欄位\: +Affected\ fields\:=影響欄位\: Show\ preview\ as\ a\ tab\ in\ entry\ editor=在條目編輯器中以分頁形式顯示預覽 Font=字型 Visual\ theme=界面主題 @@ -1056,10 +1063,6 @@ Remove\ digits=移除數字 Removes\ digits.=移除數字 -Check\ Proxy\ Setting=檢查代理設定 -Check\ connection=檢查連接 -Connection\ failed\!=連線失敗! -Connection\ successful\!=連線成功! Generate\ groups\ for\ author\ last\ names=依據作者姓氏生成群組 Regular\ expression=正規表達式 @@ -1102,3 +1105,5 @@ Select\ directory=選擇資料夾 + + diff --git a/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..75a1a176597 --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java @@ -0,0 +1,105 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.copyormovecontent.CopyOrMoveFieldContentTabViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class CopyOrMoveFieldContentTabViewModelTest { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998"); + bibDatabase = new BibDatabase(); + copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + } + + @Test + void copyValueDoesNotCopyBlankValues() { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + copyOrMoveFieldContentTabViewModel.copyValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.DATE), "YEAR field is not copied correctly to the DATE field"); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR), "YEAR field should not have changed"); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE), "DATE field should not have changed because the YEAR field is blank e.g it doesn't exist"); + } + + @Test + void swapValuesShouldNotSwapFieldValuesIfOneOfTheValuesIsBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE)); + assertEquals(Optional.empty(), entryB.getField(StandardField.YEAR)); + } + + @Test + void swapValuesShouldSwapFieldValuesIfBothValuesAreNotBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(List.of(Optional.of("2014"), Optional.of("2015")), + List.of(entryA.getField(StandardField.YEAR), entryA.getField(StandardField.DATE)), + "YEAR and DATE values didn't swap"); + } + + @Test + void moveValueShouldNotMoveValueIfToFieldIsNotBlankAndOverwriteIsNotEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(false); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + } + + @Test + void moveValueShouldMoveValueIfOverwriteIsEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + private CopyOrMoveFieldContentTabViewModel newTwoFieldsViewModel(BibEntry... selectedEntries) { + return new CopyOrMoveFieldContentTabViewModel(List.of(selectedEntries), bibDatabase, stateManager); + } +} diff --git a/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..1221580762a --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java @@ -0,0 +1,118 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.editfieldcontent.EditFieldContentViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotEquals; +import static org.mockito.Mockito.mock; + +public class EditFieldContentTabViewModelTest { + EditFieldContentViewModel editFieldContentViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, ""); + + bibDatabase = new BibDatabase(); + editFieldContentViewModel = new EditFieldContentViewModel(bibDatabase, List.of(entryA, entryB), stateManager); + } + + @Test + void clearSelectedFieldShouldClearFieldContentEvenWhenOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.YEAR)); + } + + @Test + void clearSelectedFieldShouldDoNothingWhenFieldDoesntExistOrIsEmpty() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.FILE); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.FILE)); + } + + @Test + void setFieldValueShouldNotDoAnythingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfFieldContentIsEmpty() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryB.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldDoNothingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldAppendFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("20150"), entryA.getField(StandardField.YEAR)); + } + + @Test + void getAllFieldsShouldNeverBeEmpty() { + assertNotEquals(0, editFieldContentViewModel.getAllFields().size()); + } + + @Test + void getSelectedFieldShouldHaveADefaultValue() { + assertNotEquals(null, editFieldContentViewModel.getSelectedField()); + } +} diff --git a/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java new file mode 100644 index 00000000000..bf5a246e97f --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java @@ -0,0 +1,111 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.renamefield.RenameFieldViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.field.UnknownField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class RenameFieldViewModelTest { + RenameFieldViewModel renameFieldViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014") + .withField(StandardField.AUTHOR, "Doe"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, "") + .withField(StandardField.AUTHOR, "Eddie"); + + bibDatabase = new BibDatabase(); + renameFieldViewModel = new RenameFieldViewModel(List.of(entryA, entryB), bibDatabase, stateManager); + } + + @Test + void renameFieldShouldRenameFieldIfItExist() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName("ETAD"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryA.getField(StandardField.DATE)); + + assertEquals(Optional.of("1998"), entryB.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + @Test + void renameFieldShouldDoNothingIfFieldDoNotExist() { + Field toRenameField = new UnknownField("Some_field_that_doesnt_exist"); + renameFieldViewModel.selectField(toRenameField); + renameFieldViewModel.setNewFieldName("new_field_name"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.empty(), entryA.getField(toRenameField)); + assertEquals(Optional.empty(), entryA.getField(new UnknownField("new_field_name"))); + + assertEquals(Optional.empty(), entryB.getField(toRenameField)); + assertEquals(Optional.empty(), entryB.getField(new UnknownField("new_field_name"))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameIsEmpty() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName(""); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField(""))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField(""))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameHasWhitespaceCharacters() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName("Hello, World"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField("Hello, World"))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField("Hello, World"))); + } + + @Test + void renameFieldShouldDoNothingWhenThereIsAlreadyAFieldWithTheSameNameAsNewFieldName() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName(StandardField.YEAR.getName()); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + } +} diff --git a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java index 1f1c20fc329..9fded7c7802 100644 --- a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java +++ b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java @@ -56,7 +56,7 @@ void compareWithChangedGroup() { Optional groupDiff = GroupDiff.compare(originalMetaData, newMetaData); - Optional expectedGroupDiff = Optional.of(new GroupDiff(newMetaData.getGroups().get(), originalMetaData.getGroups().get())); + Optional expectedGroupDiff = Optional.of(new GroupDiff(originalMetaData.getGroups().get(), newMetaData.getGroups().get())); assertEquals(expectedGroupDiff.get().getNewGroupRoot(), groupDiff.get().getNewGroupRoot()); assertEquals(expectedGroupDiff.get().getOriginalGroupRoot(), groupDiff.get().getOriginalGroupRoot()); diff --git a/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java new file mode 100644 index 00000000000..1b63e26cbb3 --- /dev/null +++ b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java @@ -0,0 +1,666 @@ +package org.jabref.logic.bst; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.jabref.logic.bst.util.BstCaseChangersTest; +import org.jabref.logic.bst.util.BstNameFormatterTest; +import org.jabref.logic.bst.util.BstPurifierTest; +import org.jabref.logic.bst.util.BstTextPrefixerTest; +import org.jabref.logic.bst.util.BstWidthCalculatorTest; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.types.StandardEntryType; + +import org.antlr.v4.runtime.RecognitionException; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +/** + * For additional tests see for + * + * purify: {@link BstPurifierTest} + * width: {@link BstWidthCalculatorTest} + * format.name: {@link BstNameFormatterTest} + * change.case: {@link BstCaseChangersTest} + * prefix: {@link BstTextPrefixerTest} + * + */ +class BstFunctionsTest { + @Test + public void testCompareFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test.compare } { + #5 #5 = % TRUE + #1 #2 = % FALSE + #3 #4 < % TRUE + #4 #3 < % FALSE + #4 #4 < % FALSE + #3 #4 > % FALSE + #4 #3 > % TRUE + #4 #4 > % FALSE + "H" "H" = % TRUE + "H" "Ha" = % FALSE + } + EXECUTE { test.compare } + """); + + vm.render(Collections.emptyList()); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 #1 + % 2 + #5 #2 - % 3 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(3, vm.getStack().pop()); + assertEquals(2, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctionTypeMismatch() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 "HELLO" + % Should throw exception + } + EXECUTE { test } + """); + + assertThrows(BstVMException.class, () -> vm.render(Collections.emptyList())); + } + + @Test + public void testStringOperations() throws RecognitionException { + // Test for concat (*) and add.period + BstVM vm = new BstVM(""" + FUNCTION { test } { + "H" "ello" * % Hello + "Johnny" add.period$ % Johnny. + "Johnny." add.period$ % Johnny. + "Johnny!" add.period$ % Johnny! + "Johnny?" add.period$ % Johnny? + "Johnny} }}}" add.period$ % Johnny.} + "Johnny!}" add.period$ % Johnny!} + "Johnny?}" add.period$ % Johnny?} + "Johnny.}" add.period$ % Johnny.} + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?}", vm.getStack().pop()); + assertEquals("Johnny!}", vm.getStack().pop()); + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?", vm.getStack().pop()); + assertEquals("Johnny!", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Hello", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testMissing() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { test } { title missing$ cite$ } + ITERATE { test } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Article) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "No title")); + + vm.render(testEntries); + + assertEquals("test", vm.getStack().pop()); // cite + assertEquals(BstVM.TRUE, vm.getStack().pop()); // missing title + assertEquals("canh05", vm.getStack().pop()); // cite + assertEquals(BstVM.FALSE, vm.getStack().pop()); // missing title + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testNumNames() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "Johnny Foo { and } Mary Bar" num.names$ + "Johnny Foo and Mary Bar" num.names$ + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(2, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSubstring() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "123456789" #2 #1 substring$ % 2 + "123456789" #4 global.max$ substring$ % 456789 + "123456789" #1 #9 substring$ % 123456789 + "123456789" #1 #10 substring$ % 123456789 + "123456789" #1 #99 substring$ % 123456789 + "123456789" #-7 #3 substring$ % 123 + "123456789" #-1 #1 substring$ % 9 + "123456789" #-1 #3 substring$ % 789 + "123456789" #-2 #2 substring$ % 78 + } EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("78", vm.getStack().pop()); + assertEquals("789", vm.getStack().pop()); + assertEquals("9", vm.getStack().pop()); + assertEquals("123", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("456789", vm.getStack().pop()); + assertEquals("2", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testEmpty() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + READ + STRINGS { s } + FUNCTION { test } { + s empty$ % TRUE + "" empty$ % TRUE + " " empty$ % TRUE + title empty$ % TRUE + " HALLO " empty$ % FALSE + } + ITERATE { test } + """); + List testEntry = List.of(new BibEntry(StandardEntryType.Article)); + + vm.render(testEntry); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameStatic() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { format }{ "Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin" #1 "{vv~}{ll}{, jj}{, f}?" format.name$ } + EXECUTE { format } + """); + List v = Collections.emptyList(); + + vm.render(v); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameInEntries() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { author } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { format }{ author #2 "{vv~}{ll}{, jj}{, f}?" format.name$ } + ITERATE { format } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "Jonathan Meyer and Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin")); + + vm.render(testEntries); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals("Annabi, H?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChangeCase() throws RecognitionException { + BstVM vm = new BstVM(""" + STRINGS { title } + READ + FUNCTION { format.title } { + duplicate$ empty$ + { pop$ "" } + { "t" change.case$ } + if$ + } + FUNCTION { test } { + "hello world" "u" change.case$ format.title + "Hello World" format.title + "" format.title + "{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase" "u" change.case$ format.title + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase", + vm.getStack().pop()); + assertEquals("", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testTextLength() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "hello world" text.length$ % 11 + "Hello {W}orld" text.length$ % 11 + "" text.length$ % 0 + "{A}{D}/{Cycle}" text.length$ % 8 + "{\\This is one character}" text.length$ % 1 + "{\\This {is} {one} {c{h}}aracter as well}" text.length$ % 1 + "{\\And this too" text.length$ % 1 + "These are {\\11}" text.length$ % 11 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(11, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(8, vm.getStack().pop()); + assertEquals(0, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testIntToStr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { #3 int.to.str$ #9999 int.to.str$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("9999", vm.getStack().pop()); + assertEquals("3", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToInt() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(72, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToIntIntToChr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ int.to.chr$ } + EXECUTE {test} + """); + + vm.render(Collections.emptyList()); + + assertEquals("H", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + SORT + FUNCTION { test } { type$ } + ITERATE { test } + """); + List testEntries = List.of( + new BibEntry(StandardEntryType.Article).withCitationKey("a"), + new BibEntry(StandardEntryType.Book).withCitationKey("b"), + new BibEntry(StandardEntryType.Misc).withCitationKey("c"), + new BibEntry(StandardEntryType.InProceedings).withCitationKey("d")); + + vm.render(testEntries); + + assertEquals("inproceedings", vm.getStack().pop()); + assertEquals("misc", vm.getStack().pop()); + assertEquals("book", vm.getStack().pop()); + assertEquals("article", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testCallType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { inproceedings }{ "InProceedings called on " title * } + FUNCTION { book }{ "Book called on " title * } + ITERATE { call.type$ } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.TITLE, "Test")); + + vm.render(testEntries); + + assertEquals("Book called on Test", vm.getStack().pop()); + assertEquals( + "InProceedings called on Effective work practices for floss development: A model and propositions", + vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSwap() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { a } { #3 "Hallo" swap$ } + EXECUTE { a } + """); + + List
smith
\nДля поиска в поле Автор значения Smith а в поле Заголовок значения electrical, введите\:\n
author\=smith and title\=electrical @@ -2234,7 +2299,7 @@ This\ entry\ type\ is\ intended\ for\ sources\ such\ as\ web\ sites\ which\ are\ A\ single-volume\ work\ of\ reference\ such\ as\ an\ encyclopedia\ or\ a\ dictionary.=Неделимая работа или ссылка, как энциклопедия или словарь. A\ technical\ report,\ research\ report,\ or\ white\ paper\ published\ by\ a\ university\ or\ some\ other\ institution.=Технический отчет, исследовательский отчет, или белая книга, выпущенная институтом или другим учреждением. An\ entry\ set\ is\ a\ group\ of\ entries\ which\ are\ cited\ as\ a\ single\ reference\ and\ listed\ as\ a\ single\ item\ in\ the\ bibliography.=Набор записей представляет собой группу записей, которые приводятся в виде единой ссылки и перечислены в виде одного элемента в библиографии. -Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. +Supplemental\ material\ in\ a\ "Book".\ This\ type\ is\ provided\ for\ elements\ such\ as\ prefaces,\ introductions,\ forewords,\ afterwords,\ etc.\ which\ often\ have\ a\ generic\ title\ only.=Дополнительный материал в "Книге" предназначен для таких элементов, как предисловия, введения, послесловия и т.д. Supplemental\ material\ in\ a\ "Collection".=Дополнительные материалы в "Коллекции". Supplemental\ material\ in\ a\ "Periodical".\ This\ type\ may\ be\ useful\ when\ referring\ to\ items\ such\ as\ regular\ columns,\ obituaries,\ letters\ to\ the\ editor,\ etc.\ which\ only\ have\ a\ generic\ title.=Дополнительные материалы в "Периодическом издании". Этот тип может быть полезен при обращении к таким элементам, как обычные колонки, некрологи, письма к редактору и т.д., которые имеют только общее название. A\ thesis\ written\ for\ an\ educational\ institution\ to\ satisfy\ the\ requirements\ for\ a\ degree.=Тезис, написанный для учебного заведения с целью удовлетворения требований к степени. @@ -2280,10 +2345,6 @@ Removes\ digits.=Удаляет цифры. Presets=Предустановки -Check\ Proxy\ Setting=Проверить настройки прокси -Check\ connection=Проверить подключение -Connection\ failed\!=Соединение потеряно\! -Connection\ successful\!=Соединение успешно\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Создать группы из ключевых слов в следующем поле Generate\ groups\ for\ author\ last\ names=Создание групп для фамилий авторов Regular\ expression=Регулярное выражение @@ -2305,24 +2366,40 @@ Separate\ citations=Разделить цитаты Unprotect\ terms=Незащищенные термины Generate\ a\ new\ key\ for\ imported\ entries\ (overwriting\ their\ default)=Создать новый ключ для импортированных записей (взамен существующих) +Warn\ about\ duplicates\ on\ import=Предупреждать о дубликатах при импорте Import\ and\ Export=Импорт и экспорт Custom\ DOI\ URI=Пользовательский DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=Использовать пользовательский базовый URI от DOI для доступа к материалам Cited\ on\ pages=Цитаты на страницах +Please\ move\ the\ cursor\ into\ the\ document\ text.=Пожалуйста, переместите курсов в текст документа. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=Для получения информации о зрительном положении ваших цитат, мне необходимо передвинуть курсор, однако мне не удалось его захватить. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=Я не могу осуществить вставку в текущее положение курсора. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Пожалуйста, переместите курсор в положение нового цитирования. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Пожалуйста, создайте его в документе или измените в файле\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Пожалуйста, используйте последний вариант в файле оформления ниже во избежание проблем локализации. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' символа %0 является отображаемым именем для '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' символа %0 отсутствует в документе +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=Стиль '%1' абзаца %0 является отображаемым именем для '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=Стиль '%1' абзаца %0 отсутствует в документе +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Ошибка при проверке того, записывает ли Writer изменения или содержит записанные изменения. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Невозможно выполнить при включённом [Edit]/[Track Changes]/[Record]. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Изменения, выполненные JabRef, могут привести к неожиданным взаимодействиям с записанными изменениями. +Recording\ and/or\ Recorded\ changes=Изменения записываются и/или записаны +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Сперва используйте [Edit]/[Track Changes]/[Manage], чтобы их разрешить. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=Не удается найти действительный путь сертификации к запрошенной цели (%0), все равно загрузить? Download\ operation\ canceled.=Операция загрузки отменена. @@ -2340,7 +2417,6 @@ Entry\ Type=Тип записи Entry\ types=Типы записей Field\ names=Названия полей Others=Другие -Overwrite\ existing\ field\ values=Перезаписать текущие значения полей Recommended=Рекомендованный Authors\ and\ Title=Авторы и Заголовок @@ -2371,6 +2447,8 @@ Query=Запрос Question=Вопрос Select\ directory=Выберите каталог +Fulltext\ Index=Полнотекстовое индексирование +Automatically\ index\ all\ linked\ files\ for\ fulltext\ search=Автоматически индексировать все связанные файлы для полнотекстового поиска Rebuild\ fulltext\ search\ index=Перестроить полнотекствой индекс Rebuild\ fulltext\ search\ index\ for\ current\ library?=Перестроить полнотекстовой индекс для текущей библиотеки? Rebuilding\ fulltext\ search\ index...=Перестроить полнотекствой индекс... @@ -2383,6 +2461,8 @@ Grobid\ URL=Grobid URL Remote\ services=Удаленные сервисы Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Разрешить отправку PDF-файлов и цитат в онлайн сервис JabRef (Grobid) для определения метаданных. Это приводит к улучшению результатов. +Fetcher\ cannot\ be\ tested\!=Невозможно проверить сборщик данных\! +Fetcher\ unknown\!=Неизвестный сборщик данных\! Character\ by\ character=Посимвольно Embedded=Встроенный @@ -2397,7 +2477,31 @@ Could\ not\ extract\ Metadata\ from\:\ %0=Не удалось извлечь м Search\ across\ libraries\ in\ a\ new\ window=Поиск по библиотекам в новом окне Keep\ search\ string\ across\ libraries=Сохранить строку поиска по библиотекам Keep\ dialog\ always\ on\ top=Показывать диалог поверх всех окон - - - - +Search\ results\ from\ open\ libraries=Результаты поиска в открытых библиотеках + +Add\ certificate=Добавить сертификат +Serial\ number=Серийный номер +Issuer=Издатель +Valid\ from=Действителен с +Valid\ to=Действителен до +Signature\ algorithm=Алгоритм подписи +Version=Версия + +Error\ downloading=Ошибка загрузки + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Ошибка при записи метаданных. Смотрите журнал ошибок для подробностей. +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Не удалось записать метаданные, файл %1 не найден. +Success\!\ Finished\ writing\ metadata.=Успех\! Запись метаданных завершена. + +Custom\ API\ key=Пользовательский ключ API +Check\ %0\ API\ Key\ Setting=Проверьте настройку ключа API %0 + +Set=Задать +Append=Присоединить +Rename=Переименовать +New\ field\ name=Новое имя поля +Automatic\ field\ editor=Автоматический редактор поля + +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Примечание\: Если в исходных записях отсутствуют ключевые слова, подходящие для конфигурации новой группы, то при подтверждении здесь они будут добавлены) +Assign=Назначить +Do\ not\ assign=Не назначать diff --git a/src/main/resources/l10n/JabRef_sv.properties b/src/main/resources/l10n/JabRef_sv.properties index 84779302d46..ae4d6c18d10 100644 --- a/src/main/resources/l10n/JabRef_sv.properties +++ b/src/main/resources/l10n/JabRef_sv.properties @@ -19,6 +19,7 @@ Library\ '%0'\ has\ empty\ entries.\ Do\ you\ want\ to\ delete\ them?=Biblioteke %0/%1\ entries=%0/%1 poster + %0\ matches\ the\ regular\ expression\ %1=%0 matchar det reguljära uttrycket %1 %0\ matches\ the\ term\ %1=%0 matchar termen %1 @@ -138,8 +139,6 @@ Cite\ command=Citeringskommando Clear=Rensa -Clear\ fields=Rensa fält - Open\ /\ close\ entry\ editor=Öppna / stäng postredigerare Close\ dialog=Stäng dialog @@ -618,11 +617,18 @@ Remove\ selected\ entries\ from\ this\ group=Ta bort valda poster från denna gr Remove\ group=Ta bort grupp -Remove\ group\ "%0"?=Ta bort gruppen "%0"? +Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Ta bort gruppen "%0" och dess undergrupper? -Remove\ group\ and\ subgroups=Ta bort grupp och dess undergrupper +Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. + + Remove\ link=Ta bort länk @@ -632,8 +638,6 @@ Remove\ string\ %0=Ta bort sträng %0 Removed\ group\ "%0".=Tog bort gruppen "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Tog bort gruppen "%0" och dess undergrupper. - Removed\ string=Tog bort sträng Renamed\ string=Bytte namn på sträng @@ -685,9 +689,6 @@ Select\ entry\ type=Välj posttyp Select\ file\ from\ ZIP-archive=Välj fil från ZIP-arkiv -Set\ field=Sätt fält -Set\ fields=Sätt fält - Settings=Alternativ Shortcut=Genväg @@ -718,7 +719,6 @@ Size=Storlek Skipped\ -\ No\ PDF\ linked=Hoppade över - Ingen PDF länkad Skipped\ -\ PDF\ does\ not\ exist=Hoppade över - PDF fanns ej -Skipped\ entry.=Hoppade över post. source\ edit=ändring av källkod @@ -828,8 +828,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (contains\ whitespaces).=Rad %0 Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Rad %0\: Hittade korrupt referensnyckel %1 (kommatecken saknas). Download\ from\ URL=Ladda ned från URL Rename\ field=Byt namn på fält -Rename\ field\ to=Byt namn på fält till -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Flytta innehållet i ett fält till ett fält med ett annat namn Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Kan inte använda port "%0" för fjärråtkomst; ett annat program kanske använder den. Försök med en annan port. @@ -1029,6 +1027,7 @@ Proxy\ requires\ authentication=Proxyn kräver autentisering Attention\:\ Password\ is\ stored\ in\ plain\ text\!=OBS\! Lösenordet sparas i klartext\! Clear\ connection\ settings=Rensa anslutningsinställningar + Open\ folder=Öppna mapp Export\ sort\ order=Soteringsordning vid export @@ -1370,7 +1369,6 @@ Protected\ terms\ file=Fil med skyddade termer Style\ file=Stilfil för OpenOffice/LibreOffice Open\ OpenOffice/LibreOffice\ connection=Öppna OpenOffice/LibreOffice-anslutning -You\ must\ enter\ at\ least\ one\ field\ name=Du måste ange minst ett fältnamn Non-ASCII\ encoded\ character\ found=Bokstäver som inte är ASCII-kodade hittades Toggle\ web\ search\ interface=Växla webbsökning @@ -1571,6 +1569,8 @@ Previous\ preview\ style=Föregående förhandsgranskningsstil + + diff --git a/src/main/resources/l10n/JabRef_tl.properties b/src/main/resources/l10n/JabRef_tl.properties index 0cb29b98e99..de37b3f268a 100644 --- a/src/main/resources/l10n/JabRef_tl.properties +++ b/src/main/resources/l10n/JabRef_tl.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 nag tugma sa regular na ekspresyon %1 %0\ matches\ the\ term\ %1=%0 tumugma sa termino %1 @@ -126,8 +127,6 @@ Cite\ command=Utos na nabanggit Clear=Malinaw -Clear\ fields=Malinaw na patlang - Close\ dialog=Isara ang dialog @@ -600,11 +599,18 @@ Remove\ selected\ entries\ from\ this\ group=Tanggalin ang napiling entries mula Remove\ group=Tanggaling ang grupo -Remove\ group\ "%0"?=Tanggalin ang grupo "%0"? +Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Tanggalin ang grupo "%0" at ang mababang grupo? -Remove\ group\ and\ subgroups=Tanggalin ang grupo at mga mababang grupo +Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. + + Remove\ link=Tanggalin ang link @@ -613,8 +619,6 @@ Remove\ old\ entry=Tanggalin ang lumang entry Removed\ group\ "%0".=Tanggalin ang grupo "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Na tanggal ang grupo "%0"at ang mababang mga grupo. - Removed\ string=Na tanggal ang string Renamed\ string=Palitan ulit ng pangalan ang string @@ -671,9 +675,6 @@ Select\ file\ from\ ZIP-archive=Piliin ang file mula sa na arkiba na ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Piliin ang nodes na puno para makita at matanggap o matanggihan ang pagbabago -Set\ field=Ihanda ang patlang -Set\ fields=Ihanda ang patlang - Settings=Mga settings Shortcut=Shortcut @@ -706,7 +707,6 @@ Size=Laki Skipped\ -\ No\ PDF\ linked=Nilaktawan - Walang PDF ang na linked Skipped\ -\ PDF\ does\ not\ exist=Nilaktawan - PDF ay hindi umiiral -Skipped\ entry.=Nilaktawan ang entry. source\ edit=i-edit ang pinagmulan @@ -820,10 +820,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=I-download mula sa URL Rename\ field=Palitan ang pangalan ng patlang -Append\ field=Ilagay ang field -Append\ to\ fields=Ilagay sa mga patlang -Rename\ field\ to=Palitan ang pangalan ng patlang sa -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Ilipat ang mga nilalaman ng isang patlang sa isang patlang na may ibang pangalan Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Hindi magagamit ang port %0 para sa remote na operasyon; maaaring gamitin ng isa pang application. Subukan ang pagtukoy ng isa pang port. @@ -1033,6 +1029,7 @@ Proxy\ requires\ authentication=Kinakailangan ng proxy ang pagpapatunay Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Pansin\: Ang password ay naka-imbak sa plain text\! Clear\ connection\ settings=I-clear ang mga setting ng koneksyon + Export\ sort\ order=Mag-export ng uri ng order Show\ extra\ columns=Magpakita ng mga dagdag na haligi @@ -1206,7 +1203,6 @@ Protected\ terms\ file=Protektadong mga tuntunin na file Style\ file=Estilo ng file Open\ OpenOffice/LibreOffice\ connection=Buksan ang OpenOffice/LibreOffice na koneksyon -You\ must\ enter\ at\ least\ one\ field\ name=Dapat kang magpasok ng hindi bababa sa isang pangalan ng field Non-ASCII\ encoded\ character\ found=Natagpuan ang naka-encode na karakter na hindi-ASCII Toggle\ web\ search\ interface=I-toggle ang interface ng paghahanap sa web @@ -1276,7 +1272,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Tinanggal ang l Could\ not\ retrieve\ entry\ data\ from\ '%0'.=Hindi makuha ang data mula sa entry '%0'. Entry\ from\ %0\ could\ not\ be\ parsed.=Ang entry mula sa %0 ay hindi ma-parse. Invalid\ identifier\:\ '%0'.=Di-wastong identifier\: '%0'. -This\ paper\ has\ been\ withdrawn.=Ang papel na ito ay na-withdraw. Aux\ file=Aux file Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Grupo na naglalaman ng mga entry na binanggit sa isang ibinigay na file ng TeX @@ -1364,6 +1359,8 @@ Default\ pattern=Default na pattern + + diff --git a/src/main/resources/l10n/JabRef_tr.properties b/src/main/resources/l10n/JabRef_tr.properties index 80d892c18af..fc09ebe0ce1 100644 --- a/src/main/resources/l10n/JabRef_tr.properties +++ b/src/main/resources/l10n/JabRef_tr.properties @@ -19,7 +19,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 girdi -%0\ export\ successful.\ Open\ the\ folder\ containing\ the\ saved\ file?=%0 dışa aktarım başarılı. Kaydedilen dosyayı içeren klasör açılsın mı? + %0\ matches\ the\ regular\ expression\ %1=%0 şu Düzenli İfadeyle eşleşiyor %1 @@ -151,8 +151,6 @@ Cite\ command=Alıntı komutu Clear=Sil -Clear\ fields=Alanları sil - Open\ /\ close\ entry\ editor=Girdi düzenleyicisini aç/kapat Close\ dialog=Dialoğu kapat @@ -703,11 +701,18 @@ Remove\ selected\ entries\ from\ this\ group=Seçili girdileri bu gruptan çıka Remove\ group=Grubu sil -Remove\ group\ "%0"?="%0" grubu silinsin mi? +Remove\ group\ and\ subgroups=Grubu ve altgrupları sil + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?="%0" grubu ve altgrupları silinsin mi? -Remove\ group\ and\ subgroups=Grubu ve altgrupları sil +Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. + + Remove\ link=Linki sil @@ -717,8 +722,6 @@ Remove\ string\ %0=%0 Dizgesini sil Removed\ group\ "%0".="%0" grubu silindi. -Removed\ group\ "%0"\ and\ its\ subgroups.="%0" grubu ve altgrupları silindi. - Removed\ string=Dizge silindi Renamed\ string=Dizge yeniden adlandırıldı @@ -790,9 +793,6 @@ Select\ file\ from\ ZIP-archive=ZIP arşivinden dosyayı seçiniz Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Değişiklikleri görmek ve kabul ya da reddetmek için ağaç düğümlerini seçiniz -Set\ field=Alanı ata -Set\ fields=Alanları ata - Settings=Ayarlar Shortcut=Kısayol @@ -825,7 +825,6 @@ Size=Boyut Skipped\ -\ No\ PDF\ linked=Atlandı - PDF eklenmedi Skipped\ -\ PDF\ does\ not\ exist=Atlandı - PDF mevcut değil -Skipped\ entry.=Girdi atlandı. Sort\ subgroups=Alt grupları sırala @@ -972,10 +971,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=Satır %0\: B No\ full\ text\ document\ found=Tam metin belge bulunamadı Download\ from\ URL=URL'den indir Rename\ field=Alanın adını değiştir -Append\ field=Alan ekle -Append\ to\ fields=Alanlara ekle -Rename\ field\ to=Alan adını şuna değiştir -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Alan içeriğini başka isimli bir alanın içine taşı Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Uzak operasyon için bağlantı noktası %0 kullanılamıyor; bir başka program kullanıyor olabilir. Başka bir bağlantı noktası deneyin. @@ -1243,6 +1238,15 @@ Use\ custom\ proxy\ configuration=Özelleştirilmiş vekil konfigürasyonu kulla Proxy\ requires\ authentication=Vekil sunucu kimlik denetleme gerektiriyor Attention\:\ Password\ is\ stored\ in\ plain\ text\!=Dikkat\: Parola salt metin olarak kaydedildi\! Clear\ connection\ settings=Bağlantı ayarlarını sil +Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et +Check\ connection=Bağlantıyı kontrol et +Connection\ failed\!=Bağlantı başarısız\! +Connection\ successful\!=Bağlantı başarılı\! + +SSL\ Configuration=SSL Yapılandırması +SSL\ certificate\ file=SSL sertifika dosyası +Duplicate\ Certificates=Çift nüsha Sertifikalar +You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Open\ folder=Klasörü aç Export\ sort\ order=Sıralama kriterlerini dışa aktar @@ -1703,7 +1707,6 @@ Protected\ terms\ file=Korunmuş terimler dosyası Style\ file=Stil dosyası Open\ OpenOffice/LibreOffice\ connection=OpenOffice/LibreOffice bağlantısı aç -You\ must\ enter\ at\ least\ one\ field\ name=En az bir alan adı girmelisiniz Non-ASCII\ encoded\ character\ found=ASCII koduyla kodlanmamış karakter bulundu Non-UTF-8\ encoded\ field\ found=UTF-8 ile kodlanmamış alan bulundu Toggle\ web\ search\ interface=Ağ arama arayüzünü değiştir @@ -1841,7 +1844,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=Alan içeriğin Could\ not\ retrieve\ entry\ data\ from\ '%0'.='%0' dan girdi verileri alınamadı. Entry\ from\ %0\ could\ not\ be\ parsed.=%0'dan girdi çözümlenemedi. Invalid\ identifier\:\ '%0'.=Geçersiz tanımlayıcı\: '%0 '. -This\ paper\ has\ been\ withdrawn.=Bu yayın geri çekildi. empty\ citation\ key=boş atıf anahtarı Aux\ file=Aux dosya Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=Belirli bir TeX dosyasında alıntılanmış girdileri içeren grup @@ -2324,10 +2326,6 @@ Removes\ digits.=Rakamları siler. Presets=Ön ayarlar -Check\ Proxy\ Setting=Vekil Ayarını Kontrol Et -Check\ connection=Bağlantıyı kontrol et -Connection\ failed\!=Bağlantı başarısız\! -Connection\ successful\!=Bağlantı başarılı\! Generate\ groups\ from\ keywords\ in\ the\ following\ field=Aşağıdaki alandaki anahtar sözcüklerden gruplar oluştur Generate\ groups\ for\ author\ last\ names=Yazar soyadları için gruplar oluştur Regular\ expression=Düzenli ifade @@ -2400,7 +2398,6 @@ Entry\ Type=Girdi Türü Entry\ types=Girdi türleri Field\ names=Alan adları Others=Diğerleri -Overwrite\ existing\ field\ values=Mevcut alan değerlerinin üzerine yaz Recommended=Önerilen Authors\ and\ Title=Yazarlar ve Başlık @@ -2445,6 +2442,8 @@ Grobid\ URL=Grobid URL Remote\ services=Uzak servisler Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Metaverisini belirlemek için PDF dosyalarını ve ham atıf dizgelerini bir JabRef çevrim içi servisine (Grobid) göndermeye izin ver. Bu daha iyi sonuçlara yol açar. +Fetcher\ cannot\ be\ tested\!=Getirici test edilemiyor\! +Fetcher\ unknown\!=Getirici bilinmiyor\! Character\ by\ character=Karakter karakter Embedded=Gömülü @@ -2468,10 +2467,6 @@ Valid\ from=Geçerlilik tarihi Valid\ to=Son kullanma tarihi Signature\ algorithm=İmza algoritması Version=Sürüm -SSL\ Configuration=SSL Yapılandırması -SSL\ certificate\ file=SSL sertifika dosyası -Duplicate\ Certificates=Çift nüsha Sertifikalar -You\ already\ added\ this\ certificate=Bu sertifikayı zaten eklediniz Error\ downloading=İndirmede hata @@ -2479,3 +2474,7 @@ Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=Metaverisi Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Metaverisi yazma başarısız, %1 dosyası bulunamadı. Success\!\ Finished\ writing\ metadata.=Başarılı\! Metaverisi yazma bitirildi. +Custom\ API\ key=Özel API anahtarı +Check\ %0\ API\ Key\ Setting=%0 API Anahtar Ayarlarını Kontrol Edin + + diff --git a/src/main/resources/l10n/JabRef_vi.properties b/src/main/resources/l10n/JabRef_vi.properties index 26aaa90bf04..d82df8e5422 100644 --- a/src/main/resources/l10n/JabRef_vi.properties +++ b/src/main/resources/l10n/JabRef_vi.properties @@ -13,6 +13,7 @@ + %0\ matches\ the\ regular\ expression\ %1=%0 khớp biểu thức chính tắc %1 %0\ matches\ the\ term\ %1=%0 khớp thuật ngữ %1 @@ -133,8 +134,6 @@ Cite\ command=Lệnh trích dẫn Clear=Xóa -Clear\ fields=Xóa các dữ liệu - Close\ dialog=Đóng hộp thoại @@ -612,11 +611,18 @@ Remove\ entry\ from\ import=Loại bỏ mục khỏi lệnh nhập Remove\ group=Loại bỏ nhóm -Remove\ group\ "%0"?=Loại bỏ nhóm "%0"? +Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con + + + + + Remove\ group\ "%0"\ and\ its\ subgroups?=Loại bỏ nhóm "%0" và các nhóm con của nó? -Remove\ group\ and\ subgroups=Loại bỏ nhóm và các nhóm con +Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. + + Remove\ link=Loại bỏ liên kết @@ -625,8 +631,6 @@ Remove\ old\ entry=Loại bỏ mục cũ Removed\ group\ "%0".=Đã loại bỏ nhóm "%0". -Removed\ group\ "%0"\ and\ its\ subgroups.=Đã loại bỏ nhóm "%0" và các nhóm con của nó. - Removed\ string=Đã loại bỏ chuỗi Renamed\ string=Chuỗi được đặt tên lại @@ -679,9 +683,6 @@ Select\ file\ from\ ZIP-archive=Chọn tập tin từ tập tin ZIP Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=Chọn các nốt trên sơ đồ hình cây để xem và chấp nhận hoặc từ chối thay đổi -Set\ field=Thiết lập dữ liệu -Set\ fields=Thiết lập các dữ liệu - Settings=Các thiết lập Shortcut=Phím tắt @@ -712,7 +713,6 @@ Size=Kích thước Skipped\ -\ No\ PDF\ linked=Bị bỏ qua - Không có tập tin PDF được liên kết Skipped\ -\ PDF\ does\ not\ exist=Bỏ qua - tập tin PDF không tồn tại -Skipped\ entry.=Mục bị bỏ qua. source\ edit=chỉnh sửa nguồn @@ -826,8 +826,6 @@ This\ feature\ lets\ new\ files\ be\ opened\ or\ imported\ into\ an\ already\ ru Download\ from\ URL=Kéo từ URL Rename\ field=Đổi tên dữ liệu -Rename\ field\ to=Đổi tên dữ liệu thành -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=Di chuyển nội dung của một dữ liệu sang một dữ liệu có tên khác Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=Không thể dùng cổng %0 cho lệnh chạy từ xa; một ứng dụng khác có thể đang dùng nó. Hãy thử chỉ định một cổng khác. @@ -900,6 +898,7 @@ Finished\ automatically\ setting\ external\ links.=Thiết lập tự động c + Open\ folder=Mở thư mục @@ -1135,6 +1134,8 @@ Default\ pattern=Kiểu mặc định + + diff --git a/src/main/resources/l10n/JabRef_zh_CN.properties b/src/main/resources/l10n/JabRef_zh_CN.properties index 0577de6a7b9..baca5d6da46 100644 --- a/src/main/resources/l10n/JabRef_zh_CN.properties +++ b/src/main/resources/l10n/JabRef_zh_CN.properties @@ -19,6 +19,9 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 条目 +Export\ operation\ finished\ successfully.=导出成功。 + +Reveal\ in\ File\ Explorer=在文件夹中显示 %0\ matches\ the\ regular\ expression\ %1=%0 匹配正则表达式 %1 @@ -28,7 +31,7 @@ Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (DEFAULT\ abbreviation)= Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (MEDLINE\ abbreviation)=缩写选中记录的期刊名 (MEDLINE 格式缩写) Abbreviate\ journal\ names\ of\ the\ selected\ entries\ (SHORTEST\ UNIQUE\ abbreviation)=缩写选中记录的期刊名 (最短唯一 (SHORTEST UNIQUE) 格式缩写) -Abbreviate\ names=缩写名 +Abbreviate\ names=缩写名(J. Smith) Abbreviated\ %0\ journal\ names.=缩写的 %0 期刊名称。 Abbreviation=缩写 @@ -150,8 +153,6 @@ Cite\ command=引用命令 Clear=清除 -Clear\ fields=清除字段内容 - Open\ /\ close\ entry\ editor=打开/关闭条目编辑器 Close\ dialog=关闭对话框 @@ -201,7 +202,7 @@ Could\ not\ run\ the\ 'vim'\ program.=无法运行 'vim' 程序。 Could\ not\ save\ file.=无法保存文件 Character\ encoding\ '%0'\ is\ not\ supported.=,不支持编码 '%0'。 -Create\ custom\ fields\ for\ each\ BibTeX\ entry=为每个 BibTeX 条目创建自定义字段 +Create\ custom\ fields\ for\ each\ BibTeX\ entry=Create custom fields for each BibTeX entry crossreferenced\ entries\ included=包含交叉引用的记录 @@ -265,7 +266,7 @@ Display\ help\ on\ command\ line\ options=用命令行选项显示帮助 Display\ only\ entries\ belonging\ to\ all\ selected\ groups=仅显示所选分组的交集条目 Display\ version=显示版本 -Do\ not\ abbreviate\ names=不要缩写姓名 +Do\ not\ abbreviate\ names=不缩写名(John Smith) Do\ not\ import\ entry=不导入记录 @@ -376,9 +377,9 @@ First\ select\ the\ entries\ you\ want\ keys\ to\ be\ generated\ for.=首先选 Fit\ table\ horizontally\ on\ screen=列表宽度填满屏幕宽度 Float=浮动 (结果上浮到最前) -Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式\: 选项卡\: 字段; 字段; ...(例如 General\:url;pdf;note...) +Format\:\ Tab\:field;field;...\ (e.g.\ General\:url;pdf;note...)=格式:选项卡\:字段;字段;...(例如:General\:url;pdf;note...) -Format\ of\ author\ and\ editor\ names=作者和编者的姓名格式 +Format\ of\ author\ and\ editor\ names=author和editor的姓名格式 Format\ string=格式化简写字串 Format\ used=使用的格式 @@ -492,7 +493,7 @@ Key\ bindings=热键 Key\ bindings\ changed=热键已更改 -Key\ pattern=键值表达式 +Key\ pattern=Key pattern keys\ in\ library=文献库中的键值 @@ -541,7 +542,9 @@ Moved\ group\ "%0".=移动了分组 "%0"。 Mr.\ DLib\ Privacy\ settings=Mr. DLib 隐私设置 +No\ database\ is\ open=No database is open +We\ need\ a\ database\ to\ export\ from.\ Open\ one.=We need a database to export from. Open one. No\ recommendations\ received\ from\ Mr.\ DLib\ for\ this\ entry.=没有收到 Mr. DLib 对于本条目的建议。 @@ -549,9 +552,9 @@ Error\ while\ fetching\ recommendations\ from\ Mr.DLib.=在接收 Mr. DLib 的 Name=名称 -Name\ formatter=姓名格式化器 +Name\ formatter=姓名格式化 -Natbib\ style=Natbib 格式 +Natbib\ style=Natbib样式 nested\ AUX\ files=nested AUX 文件 @@ -700,11 +703,25 @@ Remove\ selected\ entries\ from\ this\ group=从这个组里移除选定记录 Remove\ group=移除分组 -Remove\ group\ "%0"?=移除分组 "%0"? +Remove\ group\ and\ subgroups=移除分组和子分组 + +Remove\ groups\ and\ subgroups=移除分组和子分组 + +Remove\ all\ selected\ groups\ and\ keep\ their\ subgroups?=移除所有选定的分组但保留其子分组? + +Remove\ group\ "%0"\ and\ keep\ its\ subgroups?=移除分组 "%0" 但保留其子分组? + +Remove\ groups=移除分组 + +Removed\ all\ selected\ groups.=移除选中的分组 Remove\ group\ "%0"\ and\ its\ subgroups?=移除分组 "%0" 和它的子分组? -Remove\ group\ and\ subgroups=移除分组和子分组 +Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 + +Remove\ all\ selected\ groups\ and\ their\ subgroups?=移除所有选定的分组及其子分组? + +Removed\ all\ selected\ groups\ and\ their\ subgroups.=移除所有选定的分组及其子分组 Remove\ link=移除链接 @@ -714,8 +731,6 @@ Remove\ string\ %0=移除字符串 %0 Removed\ group\ "%0".=已移除分组 "%0"。 -Removed\ group\ "%0"\ and\ its\ subgroups.=已移除分组 "%0" 和它的子分组。 - Removed\ string=已移除简写字串 Renamed\ string=重命名简写字串 @@ -753,7 +768,7 @@ Review=评论 Review\ changes=复查修改 Review\ Field\ Migration=查看字段迁移 -Loading=加载中 +Loading=启动 Save=保存 Save\ all\ finished.=完成保存全部。 @@ -764,10 +779,10 @@ Save\ before\ closing=关闭前保存 Save\ library=保存 Save\ library\ as...=另存为 -Saving=保存中 +Saving=保存 Saving\ all\ libraries...=正在保存所有数据库... Saving\ library=正在保存文献库 -Library\ saved=Library saved +Library\ saved=文献库已保存 Saved\ selected\ to\ '%0'.=保存选中到 '%0'. Search=查找 @@ -787,24 +802,21 @@ Select\ file\ from\ ZIP-archive=从 ZIP-压缩包中选择文件 Select\ the\ tree\ nodes\ to\ view\ and\ accept\ or\ reject\ changes=选择树节点查看和接受/拒绝修改 -Set\ field=设置字段内容 -Set\ fields=设置字段内容 - Settings=设置 Shortcut=快捷键 Show/edit\ %0\ source=显示/编辑 %0 源代码 -Show\ 'Firstname\ Lastname'=显示 '名 (Firstname) 姓 (Lastname)' +Show\ 'Firstname\ Lastname'=名在前,姓在后(John Smith) -Show\ 'Lastname,\ Firstname'=显示 '姓 (Lastname), 名 (Firstname)' +Show\ 'Lastname,\ Firstname'=姓在前,名在后(Smith, John) Show\ BibTeX\ source\ by\ default=默认显示 BibTeX 源代码 Show\ confirmation\ dialog\ when\ deleting\ entries=删除多条记录时发出警告 -Show\ last\ names\ only=只显示“姓 (Lastname)” +Show\ last\ names\ only=仅显示姓(Smith) Show\ names\ unchanged=显示原始姓名字串 @@ -822,12 +834,17 @@ Size=大小 Skipped\ -\ No\ PDF\ linked=跳过 - 没有 PDF 链接 Skipped\ -\ PDF\ does\ not\ exist=跳过 - PDF 不存在 -Skipped\ entry.=已跳过记录 +JabRef\ skipped\ the\ entry.=JabRef 跳过了条目。 +Import\ error=导入错误 +Open\ library\ error=打开文献库出错 +Please\ check\ your\ library\ file\ for\ wrong\ syntax.=请检查您的库文件是否存在语法错误。 +SourceTab\ error=SourceTab错误 +User\ input\ via\ entry-editor\ in\ `{}bibtex\ source`\ tab\ led\ to\ failure.=通过条目编辑器的 "{}bibtex source" 标签中输入时出错。 Sort\ subgroups=排序子组 source\ edit=源代码编辑 -Special\ name\ formatters=特殊的姓名格式化器 +Special\ name\ formatters=姓名格式化 Statically\ group\ entries\ by\ manual\ assignment=手动创建静态分组 @@ -956,7 +973,7 @@ Refuse\ to\ save\ the\ library\ before\ external\ changes\ have\ been\ reviewed. Library\ protection=文献库保护 Unable\ to\ save\ library=无法保存文献库 -Citation\ key\ generator=Citationkey生成器 +Citation\ key\ generator=Citationkey Unable\ to\ open\ link.=无法打开链接。 MIME\ type=MIME 类型 @@ -969,10 +986,6 @@ Line\ %0\:\ Found\ corrupted\ citation\ key\ %1\ (comma\ missing).=第 %0 行\: No\ full\ text\ document\ found=未发现完整的文档 Download\ from\ URL=从 URL 下载 Rename\ field=重命名字段 -Append\ field=添加字段 -Append\ to\ fields=追加到字段 -Rename\ field\ to=重命名字段为 -Move\ contents\ of\ a\ field\ into\ a\ field\ with\ a\ different\ name=将某个字段的内容移动到另一个字段中 Cannot\ use\ port\ %0\ for\ remote\ operation;\ another\ application\ may\ be\ using\ it.\ Try\ specifying\ another\ port.=无法使用端口 %0 进行远程操作;该端口可能被其它应用程序占用,请使用其它端口。 @@ -1042,6 +1055,7 @@ Cite\ special=引用特殊 Extra\ information\ (e.g.\ page\ number)=额外信息(例如:页码) Manage\ citations=管理文献引用 Problem\ modifying\ citation=修改文献引用存在问题 +Problem\ collecting\ citations=Problem collecting citations Citation=文献引用 Connecting...=连接中... Select\ style=选择引用样式 @@ -1056,7 +1070,9 @@ Sync\ OpenOffice/LibreOffice\ bibliography=同步 OpenOffice/LibreOffice 参考 Select\ which\ open\ Writer\ document\ to\ work\ on=选择使用哪个打开的 Writer 文档 Connected\ to\ document=连接到文档 +Could\ not\ connect\ to\ any\ Writer\ document.\ Please\ make\ sure\ a\ document\ is\ open\ before\ using\ the\ 'Select\ Writer\ document'\ button\ to\ connect\ to\ it.=Could not connect to any Writer document. Please make sure a document is open before using the 'Select Writer document' button to connect to it. +No\ Writer\ documents\ found=No Writer documents found Insert\ a\ citation\ without\ text\ (the\ entry\ will\ appear\ in\ the\ reference\ list)=插入一条没有内容的引用(这条记录将会出现在引用列表中) Cite\ selected\ entries\ with\ extra\ information=引用包含额外信息的选中记录 @@ -1064,8 +1080,11 @@ Ensure\ that\ the\ bibliography\ is\ up-to-date=保证参考文献是最新的 Your\ OpenOffice/LibreOffice\ document\ references\ the\ citation\ key\ '%0',\ which\ could\ not\ be\ found\ in\ your\ current\ library.=您的OpenOffice/LibreOffice文档引用了 citationkey "%0",但未在当前库中找到。 +This\ operation\ requires\ a\ bibliography\ database.=This operation requires a bibliography database. +Your\ OpenOffice/LibreOffice\ document\ references\ at\ least\ %0\ citation\ keys\ which\ could\ not\ be\ found\ in\ your\ current\ library.\ Some\ of\ these\ are\ %1.=Your OpenOffice/LibreOffice document references at least %0 citation keys which could not be found in your current library. Some of these are %1. +Your\ OpenOffice/LibreOffice\ document\ references\ no\ citation\ keys\ which\ could\ also\ be\ found\ in\ your\ current\ library.=Your OpenOffice/LibreOffice document references no citation keys which could also be found in your current library. Unable\ to\ synchronize\ bibliography=无法同步参考文献 Combine\ pairs\ of\ citations\ that\ are\ separated\ by\ spaces\ only=合并仅仅由空格分隔的两段引文 @@ -1073,7 +1092,12 @@ Autodetection\ failed=自动检测失败 Please\ wait...=请稍候... Connection\ lost=连接丢失 +Could\ not\ update\ bibliography=Could not update bibliography +Not\ connected\ to\ document=Not connected to document +Problem\ combining\ cite\ markers=Problem combining cite markers +Unable\ to\ reload\ style\ file=Unable to reload style file +Problem\ during\ separating\ cite\ markers=Problem during separating cite markers Automatically\ sync\ bibliography\ when\ inserting\ citations=当插入文献引用时自动同步参考文献 Look\ up\ BibTeX\ entries\ in\ the\ active\ tab\ only=在当前标签查找 BibTeX 记录 @@ -1176,6 +1200,7 @@ Toggle\ relevance=标记为相关 Toggle\ quality\ assured=标记为质量已确认 Toggle\ print\ status=标记打印状态 Update\ keywords=更新关键词 +Problem\ connecting=Problem connecting Connection\ to\ OpenOffice/LibreOffice\ has\ been\ lost.\ Please\ make\ sure\ OpenOffice/LibreOffice\ is\ running,\ and\ try\ to\ reconnect.=与OpenOffice/LibreOffice的连接已丢失。请确保OpenOffice/LibreOffice正在运行,并尝试重新连接。 JabRef\ will\ send\ at\ least\ one\ request\ per\ entry\ to\ a\ publisher.=JabRef 将会为每个条目发送至少一个请求至发布者。 @@ -1228,6 +1253,16 @@ Use\ custom\ proxy\ configuration=使用自定义 HTTP 代理 Proxy\ requires\ authentication=代理服务器需要密码 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意\: 密码以明文形式保存\! Clear\ connection\ settings=清除连接设置 +Check\ Proxy\ Setting=检查代理设置 +Check\ connection=检查连接 +Connection\ failed\!=连接失败! +Connection\ successful\!=连接成功! + +SSL\ Configuration=SSL配置 +SSL\ configuration\ changed=SSL配置已更改 +SSL\ certificate\ file=SSL证书文件 +Duplicate\ Certificates=复制证书 +You\ already\ added\ this\ certificate=您已经添加了此证书 Open\ folder=打开文件夹 Export\ sort\ order=导出顺序 @@ -1264,6 +1299,7 @@ Please\ open\ %0\ manually.=请手动打开 %0。 The\ link\ has\ been\ copied\ to\ the\ clipboard.=链接已复制到剪贴板 Open\ %0\ file=打开文件 %0 +Could\ not\ detect\ terminal\ automatically.\ Please\ define\ a\ custom\ terminal\ in\ the\ preferences.=无法自动检测终端。请在首选项中自定义一个终端。 Cannot\ delete\ file=无法删除文件 File\ permission\ error=文件权限错误 @@ -1365,7 +1401,7 @@ Automatically\ setting\ file\ links=自动设置文件链接 Regenerating\ citation\ keys\ according\ to\ metadata=根据元数据重新生成 citationkey Regenerate\ all\ keys\ for\ the\ entries\ in\ a\ BibTeX\ file=重新生成 BibTeX 文件中所有记录的键值 Show\ debug\ level\ messages=显示调试级别消息 -Default\ library\ mode=Default library mode +Default\ library\ mode=默认文献库模式 Show\ only\ preferences\ deviating\ from\ their\ default\ value=只显示与默认值不同的首选项 default=默认 key=键值 @@ -1570,10 +1606,21 @@ Custom=自定义 Export\ cited=导出已被引用的 Unable\ to\ generate\ new\ library=无法生成新库 +The\ cursor\ is\ in\ a\ protected\ area.=The cursor is in a protected area. +JabRefException=JabRefException +No\ bibliography\ database\ is\ open\ for\ citation.=No bibliography database is open for citation. +No\ database\ is\ open\ for\ updating\ citation\ markers\ after\ citing.=No database is open for updating citation markers after citing. +No\ bibliography\ entries\ are\ selected\ for\ citation.=No bibliography entries are selected for citation. +No\ bibliography\ style\ is\ selected\ for\ citation.=No bibliography style is selected for citation. +No\ database=No database +No\ entries\ selected\ for\ citation=No entries selected for citation +Open\ one\ before\ citing.=Open one before citing. +Select\ one\ before\ citing.=Select one before citing. +Select\ some\ before\ citing.=Select some before citing. Found\ identical\ ranges=Found identical ranges Found\ overlapping\ ranges=Found overlapping ranges @@ -1621,7 +1668,7 @@ Remove\ list=删除列表 Add\ abbreviation=添加缩写 Full\ journal\ name=期刊全称 Abbreviation\ name=缩写名 -Shortest\ unique\ abbreviation=Shortest unique abbreviation +Shortest\ unique\ abbreviation=最简缩写 No\ abbreviation\ files\ loaded=缩写文件未加载 @@ -1640,6 +1687,8 @@ Issue\ report\ successful=问题报告成功 Your\ issue\ was\ reported\ in\ your\ browser.=您的问题已在浏览器中报告。 The\ log\ and\ exception\ information\ was\ copied\ to\ your\ clipboard.=日志和异常信息已复制到剪贴板中。 Please\ paste\ this\ information\ (with\ Ctrl+V)\ in\ the\ issue\ description.=请将此信息粘贴至问题描述中 (用 Ctrl + V)。 +Last\ notification=最新通知 +Check\ the\ event\ log\ to\ see\ all\ notifications=在事件日志中查看所有通知 Host=主机 Port=端口 @@ -1677,7 +1726,6 @@ Protected\ terms\ file=受保护的术语文件 Style\ file=样式文件 Open\ OpenOffice/LibreOffice\ connection=打开 OpenOffice/LibreOffice 连接 -You\ must\ enter\ at\ least\ one\ field\ name=您至少需要输入一个字段名 Non-ASCII\ encoded\ character\ found=发现Non-ASCII编码字符 Non-UTF-8\ encoded\ field\ found=未找到 UTF-8 编码的字段 Toggle\ web\ search\ interface=切换网页搜索面板 @@ -1758,7 +1806,7 @@ Unhandled\ exception\ occurred.=发生了无法处理的异常。 strings\ included=包含的字符串 Escape\ underscores=转义下划线 -Color=色彩 +Color=颜色 Please\ also\ add\ all\ steps\ to\ reproduce\ this\ issue,\ if\ possible.=如果可能的话,也请添加所有步骤以重现这个问题。 Fit\ width=适应宽度 Fit\ a\ single\ page=适应单页 @@ -1815,7 +1863,6 @@ Removes\ all\ hyphenated\ line\ breaks\ in\ the\ field\ content.=移除字段内 Could\ not\ retrieve\ entry\ data\ from\ '%0'.=无法从 "%0" 中检索条目数据。 Entry\ from\ %0\ could\ not\ be\ parsed.=无法解析 %0 中的条目。 Invalid\ identifier\:\ '%0'.=无效的标识符:'%0'。 -This\ paper\ has\ been\ withdrawn.=这篇论文已被撤回。 empty\ citation\ key=Citationkey为空 Aux\ file=Aux 文件 Group\ containing\ entries\ cited\ in\ a\ given\ TeX\ file=包含给定TeX文件中引用的条目组 @@ -1883,7 +1930,7 @@ View\ change\ log=查看变更记录 View\ event\ log=查看事件日志 Website=网站 -Override\ default\ font\ settings=跳过默认字体设置 +Override\ default\ font\ settings=覆盖默认的字体设置 Clear\ search=清除搜索 Click\ help\ to\ learn\ about\ the\ migration\ of\ pre-3.6\ databases.=点击帮助了解 pre-3.6 数据库的迁移情况。 @@ -1993,47 +2040,47 @@ Import\ entries\ from\ LaTeX\ files=Import entries from LaTeX files Import\ new\ entries=Import new entries Group\ color=分组颜色 -Columns=Columns +Columns=列 File\ type=File type IEEE=IEEE Internal=Internal Special=Special Remove\ column=Remove column -Add\ custom\ column=Add custom column -Update\ to\ current\ column\ order=Update to current column order -Sort\ column\ one\ step\ upwards=Sort column one step upwards -Sort\ column\ one\ step\ downwards=Sort column one step downwards +Add\ custom\ column=添加自定义列 +Update\ to\ current\ column\ order=更新到当前的列顺序 +Sort\ column\ one\ step\ upwards=上移一位 +Sort\ column\ one\ step\ downwards=下移一位 List\ must\ not\ be\ empty.=List must not be empty. -Order=Order +Order=排序 -Add\ field\ to\ filter\ list=Add field to filter list +Add\ field\ to\ filter\ list=添加字段到过滤列表 Add\ formatter\ to\ list=Add formatter to list Filter\ List=Filter List Open\ files...=打开文件... Affected\ fields\:=补全字段 -Show\ preview\ as\ a\ tab\ in\ entry\ editor=Show preview as a tab in entry editor -Font=Font -Visual\ theme=Visual theme +Show\ preview\ as\ a\ tab\ in\ entry\ editor=在条目编辑器中以标签形式显示预览 +Font=字体 +Visual\ theme=主题风格 Light\ theme=明亮主题 Dark\ theme=暗色主题 Custom\ theme=自定义 Overwrite\ existing\ keys=覆盖该字段的内容 -Key\ patterns=不同类型条目的键值风格 +Key\ patterns=Citationkey样式 Font\ settings=Font settings Please\ specify\ a\ css\ theme\ file.=请指定一个 css 主题文件。 You\ must\ enter\ an\ integer\ value\ higher\ than\ 8.=You must enter an integer value higher than 8. Letters\ after\ duplicate\ generated\ keys=为重复的Citationkey添加字母后缀 -Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 A (a, b, ...) -Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 B (b, c, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ A\ (a,\ b,\ ...)=在重复的键值后添加字母 a (a, b, ...) +Start\ on\ second\ duplicate\ key\ with\ letter\ B\ (b,\ c,\ ...)=在重复的键值后添加字母 b (b, c, ...) Always\ add\ letter\ (a,\ b,\ ...)\ to\ generated\ keys=在生成键值时总是添加字母 (a, b, ...) -Default\ pattern=默认模式 +Default\ pattern=Default pattern Reset\ %s\ to\ default\ value=Reset %s to default value Library\ mode=Library mode Reset\ to\ recommended=Reset to recommended Remove\ all=Remove all -Reset\ All=Reset All +Reset\ All=全部重置 Linked\ identifiers=Linked identifiers insert\ entries=insert entries @@ -2056,13 +2103,14 @@ Explicit\ selection=手动选择 Searching\ for\ a\ keyword=指定字段的关键词 Free\ search\ expression=检索语句 Specified\ keywords=字段内容 -Cited\ entries=Cited entries +Cited\ entries=根据AUX文件 Search\ term\ is\ empty.=Search term is empty. Invalid\ regular\ expression.=无效的正则表达式 Please\ provide\ a\ valid\ aux\ file.=请提供一个有效的AUX文件 Keyword\ delimiter=Keyword delimiter Hierarchical\ keyword\ delimiter=Hierarchical keyword delimiter Escape\ ampersands=转义逻辑"与"符号 +Escape\ dollar\ sign=Escape dollar sign Hint\:\n\nTo\ search\ all\ fields\ for\ Smith,\ enter\:\nsmith\n\nTo\ search\ the\ field\ author\ for\ Smith\ and\ the\ field\ title\ for\ electrical,\ enter\:\nauthor\=Smith\ and\ title\=electrical=Hint\:\n\nTo search all fields for Smith, enter\:\nsmith\n\nTo search the field author for Smith and the field title for electrical, enter\:\nauthor\=Smith and title\=electrical @@ -2272,7 +2320,7 @@ Reveal\ in\ file\ explorer=Reveal in file explorer Autolink\ files=自动链接文件 -Custom\ editor\ tabs=条目编辑器的标签 +Custom\ editor\ tabs=条目编辑器 Custom\ export\ formats=自定义导出格式 Custom\ import\ formats=自定义导入格式 @@ -2283,7 +2331,7 @@ Customized\ preview\ style=自定义预览样式 Next\ preview\ style=下一个预览样式 Previous\ preview\ style=上一个预览样式 -(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=( Note\: Press return to commit changes in the table\! ) +(\ Note\:\ Press\ return\ to\ commit\ changes\ in\ the\ table\!\ )=(注意:按回车键以确认更改) Reset=Reset New\ inproceedings=新的inproceedings Reset\ entry\ types\ and\ fields\ to\ defaults=Reset entry types and fields to defaults @@ -2297,10 +2345,6 @@ Removes\ digits.=Removes digits. Presets=默认 -Check\ Proxy\ Setting=检查代理设置 -Check\ connection=检查连接 -Connection\ failed\!=连接失败! -Connection\ successful\!=连接成功! Generate\ groups\ from\ keywords\ in\ the\ following\ field=根据以下字段的关键字生成组 Generate\ groups\ for\ author\ last\ names=根据作者的姓氏生成组 Regular\ expression=正则表达式 @@ -2328,19 +2372,34 @@ Custom\ DOI\ URI=自定义 DOI URI Use\ custom\ DOI\ base\ URI\ for\ article\ access=使用基于DOI的URI访问文章 Cited\ on\ pages=Cited on pages +Please\ move\ the\ cursor\ into\ the\ document\ text.=Please move the cursor into the document text. +To\ get\ the\ visual\ positions\ of\ your\ citations\ I\ need\ to\ move\ the\ cursor\ around,\ but\ could\ not\ get\ it.=To get the visual positions of your citations I need to move the cursor around, but could not get it. +I\ cannot\ insert\ to\ the\ cursors\ current\ location.=I cannot insert to the cursors current location. +Please\ move\ the\ cursor\ to\ the\ location\ for\ the\ new\ citation.=Please move the cursor to the location for the new citation. +Please\ create\ it\ in\ the\ document\ or\ change\ in\ the\ file\:=Please create it in the document or change in the file\: +Please\ use\ the\ latter\ in\ the\ style\ file\ below\ to\ avoid\ localization\ problems.=Please use the latter in the style file below to avoid localization problems. +The\ %0\ character\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 character style '%1' is a display name for '%2'. +The\ %0\ character\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 character style '%1' is missing from the document +The\ %0\ paragraph\ style\ '%1'\ is\ a\ display\ name\ for\ '%2'.=The %0 paragraph style '%1' is a display name for '%2'. +The\ %0\ paragraph\ style\ '%1'\ is\ missing\ from\ the\ document=The %0 paragraph style '%1' is missing from the document +Error\ while\ checking\ if\ Writer\ is\ recording\ changes\ or\ has\ recorded\ changes.=Error while checking if Writer is recording changes or has recorded changes. +Cannot\ work\ with\ [Edit]/[Track\ Changes]/[Record]\ turned\ on.=Cannot work with [Edit]/[Track Changes]/[Record] turned on. +Changes\ by\ JabRef\ could\ result\ in\ unexpected\ interactions\ with\ recorded\ changes.=Changes by JabRef could result in unexpected interactions with recorded changes. +Recording\ and/or\ Recorded\ changes=Recording and/or Recorded changes +Use\ [Edit]/[Track\ Changes]/[Manage]\ to\ resolve\ them\ first.=Use [Edit]/[Track Changes]/[Manage] to resolve them first. Unable\ to\ find\ valid\ certification\ path\ to\ requested\ target(%0),\ download\ anyway?=无法找到请求目标(%0)的有效证书路径,仍然下载? Download\ operation\ canceled.=下载操作已取消。 @@ -2358,7 +2417,6 @@ Entry\ Type=条目类型 Entry\ types=条目类型 Field\ names=字段名 Others=其他 -Overwrite\ existing\ field\ values=覆盖现有字段内容 Recommended=推荐 Authors\ and\ Title=作者和标题 @@ -2401,8 +2459,10 @@ Found\ matches\ in\ Annotations\:=在注释中找到匹配项: Grobid\ URL=Grobid URL Remote\ services=远程服务 -Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=Allow sending PDF files and raw citation strings to a JabRef online service (Grobid) to determine Metadata. This produces better results. +Allow\ sending\ PDF\ files\ and\ raw\ citation\ strings\ to\ a\ JabRef\ online\ service\ (Grobid)\ to\ determine\ Metadata.\ This\ produces\ better\ results.=允许将 PDF 文件和原始引用发送到JabRef在线服务(Grobid)以获取元数据 +Fetcher\ cannot\ be\ tested\!=Fetcher cannot be tested\! +Fetcher\ unknown\!=Fetcher unknown\! Character\ by\ character=Character by character Embedded=内嵌 @@ -2419,6 +2479,29 @@ Keep\ search\ string\ across\ libraries=在所有库中保留搜索语句 Keep\ dialog\ always\ on\ top=保持窗口位于最前面 Search\ results\ from\ open\ libraries=Search results from open libraries +Add\ certificate=添加证书 +Serial\ number=序号 +Issuer=签发者 +Valid\ from=起始有效期 +Valid\ to=截至有效期 +Signature\ algorithm=签名算法 +Version=版本 + +Error\ downloading=下载出错 + +Error\ while\ writing\ metadata.\ See\ the\ error\ log\ for\ details.=写入元数据时出错。详细信息请参阅错误日志。 +Failed\ to\ write\ metadata,\ file\ %1\ not\ found.=Failed to write metadata, file %1 not found. +Success\!\ Finished\ writing\ metadata.=Success\! Finished writing metadata. +Custom\ API\ key=自定义API +Check\ %0\ API\ Key\ Setting=Check %0 API Key Setting +Set=设定 +Append=附加 +Rename=重命名 +New\ field\ name=新的字段名 +Automatic\ field\ editor=自动化条目编辑 +(Note\:\ If\ original\ entries\ lack\ keywords\ to\ qualify\ for\ the\ new\ group\ configuration,\ confirming\ here\ will\ add\ them)=(Note\: If original entries lack keywords to qualify for the new group configuration, confirming here will add them) +Assign=Assign +Do\ not\ assign=Do not assign diff --git a/src/main/resources/l10n/JabRef_zh_TW.properties b/src/main/resources/l10n/JabRef_zh_TW.properties index 588f8dd4d77..288532048c0 100644 --- a/src/main/resources/l10n/JabRef_zh_TW.properties +++ b/src/main/resources/l10n/JabRef_zh_TW.properties @@ -15,6 +15,7 @@ Unable\ to\ monitor\ file\ changes.\ Please\ close\ files\ and\ processes\ and\ %0/%1\ entries=%0/%1 條目 + %0\ matches\ the\ regular\ expression\ %1=%0 符合正規表達式 %1 %0\ matches\ the\ term\ %1=%0 符合字詞 %1 @@ -142,8 +143,6 @@ Cite\ command=引用指令 Clear=清除 -Clear\ fields=清除欄位 - Close\ dialog=關閉對話視窗 @@ -579,6 +578,13 @@ Remove\ group=移除群組 + + + + + + + Remove\ link=移除連結 Remove\ old\ entry=移除舊有條目 @@ -588,7 +594,6 @@ Remove\ string\ %0=移除字串 %0 - Replace=取代 Replace\ With\:=取代為: Find\:=尋找: @@ -627,7 +632,6 @@ Select\ all=全選 - Settings=設定 Shortcut=快速鍵 @@ -728,8 +732,6 @@ Unable\ to\ save\ library=無法儲存文獻倉庫 Rename\ field=重新命名欄位 -Append\ field=添加欄位 -Append\ to\ fields=添加到欄位 Autosave\ local\ libraries=自動儲存本機文獻倉庫 @@ -806,6 +808,11 @@ Proxy\ configuration=代理伺服器設定 Use\ custom\ proxy\ configuration=使用自訂代理伺服器 Proxy\ requires\ authentication=代理伺服器需要認證 Attention\:\ Password\ is\ stored\ in\ plain\ text\!=注意:密碼將以明文形式存放! +Check\ Proxy\ Setting=檢查代理設定 +Check\ connection=檢查連接 +Connection\ failed\!=連線失敗! +Connection\ successful\!=連線成功! + Export\ sort\ order=匯出順序 @@ -984,7 +991,7 @@ Set\ rank\ to\ five=設定評分為 5 級 Order=順序 -Affected\ fields\:=影響欄位\: +Affected\ fields\:=影響欄位\: Show\ preview\ as\ a\ tab\ in\ entry\ editor=在條目編輯器中以分頁形式顯示預覽 Font=字型 Visual\ theme=界面主題 @@ -1056,10 +1063,6 @@ Remove\ digits=移除數字 Removes\ digits.=移除數字 -Check\ Proxy\ Setting=檢查代理設定 -Check\ connection=檢查連接 -Connection\ failed\!=連線失敗! -Connection\ successful\!=連線成功! Generate\ groups\ for\ author\ last\ names=依據作者姓氏生成群組 Regular\ expression=正規表達式 @@ -1102,3 +1105,5 @@ Select\ directory=選擇資料夾 + + diff --git a/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..75a1a176597 --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/CopyOrMoveFieldContentTabViewModelTest.java @@ -0,0 +1,105 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.copyormovecontent.CopyOrMoveFieldContentTabViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class CopyOrMoveFieldContentTabViewModelTest { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998"); + bibDatabase = new BibDatabase(); + copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + } + + @Test + void copyValueDoesNotCopyBlankValues() { + CopyOrMoveFieldContentTabViewModel copyOrMoveFieldContentTabViewModel = newTwoFieldsViewModel(entryA, entryB); + + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + copyOrMoveFieldContentTabViewModel.copyValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.DATE), "YEAR field is not copied correctly to the DATE field"); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR), "YEAR field should not have changed"); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE), "DATE field should not have changed because the YEAR field is blank e.g it doesn't exist"); + } + + @Test + void swapValuesShouldNotSwapFieldValuesIfOneOfTheValuesIsBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.DATE)); + assertEquals(Optional.empty(), entryB.getField(StandardField.YEAR)); + } + + @Test + void swapValuesShouldSwapFieldValuesIfBothValuesAreNotBlank() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.swapValues(); + + assertEquals(List.of(Optional.of("2014"), Optional.of("2015")), + List.of(entryA.getField(StandardField.YEAR), entryA.getField(StandardField.DATE)), + "YEAR and DATE values didn't swap"); + } + + @Test + void moveValueShouldNotMoveValueIfToFieldIsNotBlankAndOverwriteIsNotEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(false); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + } + + @Test + void moveValueShouldMoveValueIfOverwriteIsEnabled() { + copyOrMoveFieldContentTabViewModel.fromFieldProperty().set(StandardField.DATE); + copyOrMoveFieldContentTabViewModel.toFieldProperty().set(StandardField.YEAR); + copyOrMoveFieldContentTabViewModel.overwriteFieldContentProperty().set(true); + + copyOrMoveFieldContentTabViewModel.moveValue(); + + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + private CopyOrMoveFieldContentTabViewModel newTwoFieldsViewModel(BibEntry... selectedEntries) { + return new CopyOrMoveFieldContentTabViewModel(List.of(selectedEntries), bibDatabase, stateManager); + } +} diff --git a/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java new file mode 100644 index 00000000000..1221580762a --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/EditFieldContentTabViewModelTest.java @@ -0,0 +1,118 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.editfieldcontent.EditFieldContentViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotEquals; +import static org.mockito.Mockito.mock; + +public class EditFieldContentTabViewModelTest { + EditFieldContentViewModel editFieldContentViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, ""); + + bibDatabase = new BibDatabase(); + editFieldContentViewModel = new EditFieldContentViewModel(bibDatabase, List.of(entryA, entryB), stateManager); + } + + @Test + void clearSelectedFieldShouldClearFieldContentEvenWhenOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.YEAR)); + } + + @Test + void clearSelectedFieldShouldDoNothingWhenFieldDoesntExistOrIsEmpty() { + editFieldContentViewModel.selectedFieldProperty().set(StandardField.FILE); + editFieldContentViewModel.clearSelectedField(); + + assertEquals(Optional.empty(), entryA.getField(StandardField.FILE)); + } + + @Test + void setFieldValueShouldNotDoAnythingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryA.getField(StandardField.YEAR)); + } + + @Test + void setFieldValueShouldSetFieldValueIfFieldContentIsEmpty() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("2001"); + editFieldContentViewModel.setFieldValue(); + + assertEquals(Optional.of("2001"), entryB.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldDoNothingIfOverwriteFieldContentIsNotEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(false); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + } + + @Test + void appendToFieldValueShouldAppendFieldValueIfOverwriteFieldContentIsEnabled() { + editFieldContentViewModel.overwriteFieldContentProperty().set(true); + editFieldContentViewModel.selectedFieldProperty().set(StandardField.YEAR); + editFieldContentViewModel.fieldValueProperty().set("0"); + editFieldContentViewModel.appendToFieldValue(); + + assertEquals(Optional.of("20150"), entryA.getField(StandardField.YEAR)); + } + + @Test + void getAllFieldsShouldNeverBeEmpty() { + assertNotEquals(0, editFieldContentViewModel.getAllFields().size()); + } + + @Test + void getSelectedFieldShouldHaveADefaultValue() { + assertNotEquals(null, editFieldContentViewModel.getSelectedField()); + } +} diff --git a/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java new file mode 100644 index 00000000000..bf5a246e97f --- /dev/null +++ b/src/test/java/org/jabref/gui/edit/RenameFieldViewModelTest.java @@ -0,0 +1,111 @@ +package org.jabref.gui.edit; + +import java.util.List; +import java.util.Optional; + +import org.jabref.gui.StateManager; +import org.jabref.gui.edit.automaticfiededitor.renamefield.RenameFieldViewModel; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.Field; +import org.jabref.model.entry.field.FieldFactory; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.field.UnknownField; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.mock; + +public class RenameFieldViewModelTest { + RenameFieldViewModel renameFieldViewModel; + BibEntry entryA; + BibEntry entryB; + + BibDatabase bibDatabase; + + StateManager stateManager = mock(StateManager.class); + + @BeforeEach + void setup() { + entryA = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.YEAR, "2015") + .withField(StandardField.DATE, "2014") + .withField(StandardField.AUTHOR, "Doe"); + + entryB = new BibEntry(BibEntry.DEFAULT_TYPE) + .withField(StandardField.DATE, "1998") + .withField(StandardField.YEAR, "") + .withField(StandardField.AUTHOR, "Eddie"); + + bibDatabase = new BibDatabase(); + renameFieldViewModel = new RenameFieldViewModel(List.of(entryA, entryB), bibDatabase, stateManager); + } + + @Test + void renameFieldShouldRenameFieldIfItExist() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName("ETAD"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryA.getField(StandardField.DATE)); + + assertEquals(Optional.of("1998"), entryB.getField(FieldFactory.parseField("ETAD"))); + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + } + + @Test + void renameFieldShouldDoNothingIfFieldDoNotExist() { + Field toRenameField = new UnknownField("Some_field_that_doesnt_exist"); + renameFieldViewModel.selectField(toRenameField); + renameFieldViewModel.setNewFieldName("new_field_name"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.empty(), entryA.getField(toRenameField)); + assertEquals(Optional.empty(), entryA.getField(new UnknownField("new_field_name"))); + + assertEquals(Optional.empty(), entryB.getField(toRenameField)); + assertEquals(Optional.empty(), entryB.getField(new UnknownField("new_field_name"))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameIsEmpty() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName(""); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField(""))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField(""))); + } + + @Test + void renameFieldShouldNotDoAnythingIfTheNewFieldNameHasWhitespaceCharacters() { + renameFieldViewModel.selectField(StandardField.AUTHOR); + renameFieldViewModel.setNewFieldName("Hello, World"); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("Doe"), entryA.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryA.getField(FieldFactory.parseField("Hello, World"))); + + assertEquals(Optional.of("Eddie"), entryB.getField(StandardField.AUTHOR)); + assertEquals(Optional.empty(), entryB.getField(FieldFactory.parseField("Hello, World"))); + } + + @Test + void renameFieldShouldDoNothingWhenThereIsAlreadyAFieldWithTheSameNameAsNewFieldName() { + renameFieldViewModel.selectField(StandardField.DATE); + renameFieldViewModel.setNewFieldName(StandardField.YEAR.getName()); + renameFieldViewModel.renameField(); + + assertEquals(Optional.of("2014"), entryA.getField(StandardField.DATE)); + assertEquals(Optional.of("2015"), entryA.getField(StandardField.YEAR)); + + assertEquals(Optional.empty(), entryB.getField(StandardField.DATE)); + assertEquals(Optional.of("1998"), entryB.getField(StandardField.YEAR)); + } +} diff --git a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java index 1f1c20fc329..9fded7c7802 100644 --- a/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java +++ b/src/test/java/org/jabref/logic/bibtex/comparator/GroupDiffTest.java @@ -56,7 +56,7 @@ void compareWithChangedGroup() { Optional groupDiff = GroupDiff.compare(originalMetaData, newMetaData); - Optional expectedGroupDiff = Optional.of(new GroupDiff(newMetaData.getGroups().get(), originalMetaData.getGroups().get())); + Optional expectedGroupDiff = Optional.of(new GroupDiff(originalMetaData.getGroups().get(), newMetaData.getGroups().get())); assertEquals(expectedGroupDiff.get().getNewGroupRoot(), groupDiff.get().getNewGroupRoot()); assertEquals(expectedGroupDiff.get().getOriginalGroupRoot(), groupDiff.get().getOriginalGroupRoot()); diff --git a/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java new file mode 100644 index 00000000000..1b63e26cbb3 --- /dev/null +++ b/src/test/java/org/jabref/logic/bst/BstFunctionsTest.java @@ -0,0 +1,666 @@ +package org.jabref.logic.bst; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.jabref.logic.bst.util.BstCaseChangersTest; +import org.jabref.logic.bst.util.BstNameFormatterTest; +import org.jabref.logic.bst.util.BstPurifierTest; +import org.jabref.logic.bst.util.BstTextPrefixerTest; +import org.jabref.logic.bst.util.BstWidthCalculatorTest; +import org.jabref.model.database.BibDatabase; +import org.jabref.model.entry.BibEntry; +import org.jabref.model.entry.field.StandardField; +import org.jabref.model.entry.types.StandardEntryType; + +import org.antlr.v4.runtime.RecognitionException; +import org.junit.jupiter.api.Test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +/** + * For additional tests see for + * + * purify: {@link BstPurifierTest} + * width: {@link BstWidthCalculatorTest} + * format.name: {@link BstNameFormatterTest} + * change.case: {@link BstCaseChangersTest} + * prefix: {@link BstTextPrefixerTest} + * + */ +class BstFunctionsTest { + @Test + public void testCompareFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test.compare } { + #5 #5 = % TRUE + #1 #2 = % FALSE + #3 #4 < % TRUE + #4 #3 < % FALSE + #4 #4 < % FALSE + #3 #4 > % FALSE + #4 #3 > % TRUE + #4 #4 > % FALSE + "H" "H" = % TRUE + "H" "Ha" = % FALSE + } + EXECUTE { test.compare } + """); + + vm.render(Collections.emptyList()); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctions() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 #1 + % 2 + #5 #2 - % 3 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(3, vm.getStack().pop()); + assertEquals(2, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testArithmeticFunctionTypeMismatch() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + #1 "HELLO" + % Should throw exception + } + EXECUTE { test } + """); + + assertThrows(BstVMException.class, () -> vm.render(Collections.emptyList())); + } + + @Test + public void testStringOperations() throws RecognitionException { + // Test for concat (*) and add.period + BstVM vm = new BstVM(""" + FUNCTION { test } { + "H" "ello" * % Hello + "Johnny" add.period$ % Johnny. + "Johnny." add.period$ % Johnny. + "Johnny!" add.period$ % Johnny! + "Johnny?" add.period$ % Johnny? + "Johnny} }}}" add.period$ % Johnny.} + "Johnny!}" add.period$ % Johnny!} + "Johnny?}" add.period$ % Johnny?} + "Johnny.}" add.period$ % Johnny.} + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?}", vm.getStack().pop()); + assertEquals("Johnny!}", vm.getStack().pop()); + assertEquals("Johnny.}", vm.getStack().pop()); + assertEquals("Johnny?", vm.getStack().pop()); + assertEquals("Johnny!", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Johnny.", vm.getStack().pop()); + assertEquals("Hello", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testMissing() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { test } { title missing$ cite$ } + ITERATE { test } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Article) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "No title")); + + vm.render(testEntries); + + assertEquals("test", vm.getStack().pop()); // cite + assertEquals(BstVM.TRUE, vm.getStack().pop()); // missing title + assertEquals("canh05", vm.getStack().pop()); // cite + assertEquals(BstVM.FALSE, vm.getStack().pop()); // missing title + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testNumNames() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "Johnny Foo { and } Mary Bar" num.names$ + "Johnny Foo and Mary Bar" num.names$ + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(2, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSubstring() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "123456789" #2 #1 substring$ % 2 + "123456789" #4 global.max$ substring$ % 456789 + "123456789" #1 #9 substring$ % 123456789 + "123456789" #1 #10 substring$ % 123456789 + "123456789" #1 #99 substring$ % 123456789 + "123456789" #-7 #3 substring$ % 123 + "123456789" #-1 #1 substring$ % 9 + "123456789" #-1 #3 substring$ % 789 + "123456789" #-2 #2 substring$ % 78 + } EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("78", vm.getStack().pop()); + assertEquals("789", vm.getStack().pop()); + assertEquals("9", vm.getStack().pop()); + assertEquals("123", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("123456789", vm.getStack().pop()); + assertEquals("456789", vm.getStack().pop()); + assertEquals("2", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testEmpty() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + READ + STRINGS { s } + FUNCTION { test } { + s empty$ % TRUE + "" empty$ % TRUE + " " empty$ % TRUE + title empty$ % TRUE + " HALLO " empty$ % FALSE + } + ITERATE { test } + """); + List testEntry = List.of(new BibEntry(StandardEntryType.Article)); + + vm.render(testEntry); + + assertEquals(BstVM.FALSE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(BstVM.TRUE, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameStatic() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { format }{ "Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin" #1 "{vv~}{ll}{, jj}{, f}?" format.name$ } + EXECUTE { format } + """); + List v = Collections.emptyList(); + + vm.render(v); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testFormatNameInEntries() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { author } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { format }{ author #2 "{vv~}{ll}{, jj}{, f}?" format.name$ } + ITERATE { format } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.AUTHOR, "Jonathan Meyer and Charles Louis Xavier Joseph de la Vall{\\'e}e Poussin")); + + vm.render(testEntries); + + assertEquals("de~la Vall{\\'e}e~Poussin, C.~L. X.~J?", vm.getStack().pop()); + assertEquals("Annabi, H?", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChangeCase() throws RecognitionException { + BstVM vm = new BstVM(""" + STRINGS { title } + READ + FUNCTION { format.title } { + duplicate$ empty$ + { pop$ "" } + { "t" change.case$ } + if$ + } + FUNCTION { test } { + "hello world" "u" change.case$ format.title + "Hello World" format.title + "" format.title + "{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase" "u" change.case$ format.title + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("{A}{D}/{C}ycle: {I}{B}{M}'s {F}ramework for {A}pplication {D}evelopment and {C}ase", + vm.getStack().pop()); + assertEquals("", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals("Hello world", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testTextLength() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { + "hello world" text.length$ % 11 + "Hello {W}orld" text.length$ % 11 + "" text.length$ % 0 + "{A}{D}/{Cycle}" text.length$ % 8 + "{\\This is one character}" text.length$ % 1 + "{\\This {is} {one} {c{h}}aracter as well}" text.length$ % 1 + "{\\And this too" text.length$ % 1 + "These are {\\11}" text.length$ % 11 + } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(11, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(1, vm.getStack().pop()); + assertEquals(8, vm.getStack().pop()); + assertEquals(0, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(11, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testIntToStr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { #3 int.to.str$ #9999 int.to.str$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals("9999", vm.getStack().pop()); + assertEquals("3", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToInt() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ } + EXECUTE { test } + """); + + vm.render(Collections.emptyList()); + + assertEquals(72, vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testChrToIntIntToChr() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { test } { "H" chr.to.int$ int.to.chr$ } + EXECUTE {test} + """); + + vm.render(Collections.emptyList()); + + assertEquals("H", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + SORT + FUNCTION { test } { type$ } + ITERATE { test } + """); + List testEntries = List.of( + new BibEntry(StandardEntryType.Article).withCitationKey("a"), + new BibEntry(StandardEntryType.Book).withCitationKey("b"), + new BibEntry(StandardEntryType.Misc).withCitationKey("c"), + new BibEntry(StandardEntryType.InProceedings).withCitationKey("d")); + + vm.render(testEntries); + + assertEquals("inproceedings", vm.getStack().pop()); + assertEquals("misc", vm.getStack().pop()); + assertEquals("book", vm.getStack().pop()); + assertEquals("article", vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testCallType() throws RecognitionException { + BstVM vm = new BstVM(""" + ENTRY { title } { } { } + FUNCTION { presort } { cite$ 'sort.key$ := } + ITERATE { presort } + READ + SORT + FUNCTION { inproceedings }{ "InProceedings called on " title * } + FUNCTION { book }{ "Book called on " title * } + ITERATE { call.type$ } + """); + List testEntries = List.of( + BstVMTest.defaultTestEntry(), + new BibEntry(StandardEntryType.Book) + .withCitationKey("test") + .withField(StandardField.TITLE, "Test")); + + vm.render(testEntries); + + assertEquals("Book called on Test", vm.getStack().pop()); + assertEquals( + "InProceedings called on Effective work practices for floss development: A model and propositions", + vm.getStack().pop()); + assertEquals(0, vm.getStack().size()); + } + + @Test + public void testSwap() throws RecognitionException { + BstVM vm = new BstVM(""" + FUNCTION { a } { #3 "Hallo" swap$ } + EXECUTE { a } + """); + + List