From bd7219f1a02ce6c0441a170288b9192bd241f4e7 Mon Sep 17 00:00:00 2001 From: Loay Ghreeb Date: Wed, 23 Oct 2024 14:04:32 +0300 Subject: [PATCH] Searching with Postgres (#11803) * Convert to record * Add embedded-postgres dependency * Setup embedded postgre server * Indexing entries with Postgres * Remove Lucene bib fields indexer * Use PreparedStatement to fix escaping characters * Rename LuceneManager to IndexManager * Begin to implement "new" search syntax Co-authored-by: Loay Ghreeb <52158423+LoayGhreeb@users.noreply.github.com> Co-authored-by: Carl Christian Snethlage <50491877+calixtus@users.noreply.github.com> * Some new operators * Change log level of EmbeddedPostgres * Handle search flags * Fix default field * Fix handling of anyfield (and add "any" as alias) * Openrewrite... * More test cases * Remove non-covered libraries * Use LIKE syntax as default instead of regex Disabled exact match * Update module-info.java * Create "query" package * Postgres searcher * Return back the exact match operator * checkstyle * Add link * WIP * Fix compilation * WIP * Intermediate result Co-authored-by: Loay Ghreeb <52158423+LoayGhreeb@users.noreply.github.com> * Query should be OK Co-authored-by: Loay Ghreeb <52158423+LoayGhreeb@users.noreply.github.com> * Indexing of split values Co-authored-by: Loay Ghreeb <52158423+LoayGhreeb@users.noreply.github.com> * Fix tests compile * Use first name Last name for authors * Refactor SQL query visitor * Adapt tests * Use join with EXACT_MATCH only * Update to Postgres 17 * Attempt to use sub-queries with CTEs * Fix CTEs sub-queries and grouping TODO: EXCAT_MATCH to search in split table * group matches by entry_id * Use NOT IN for negation queries * Fix unary NOT operator * Use split values table for EXACT_MATCH queries * Prepare for linked files index (full-text) * Prepare linked files tables * Fix searching * Use multi column index * fix merged module issues * Fix update event * Remove postgres linked files indexer * Remove and insert field on update event * Remove search score column * Update search groups matches * Remove search_score from tale preferences * Migrate search groups flags to new syntax * Localization * Fix dialog message * Ignores groups field from default searches Fixes https://github.com/JabRef/jabref/issues/7996 * Use TYPE_HEADER field for entrytype * Search to Lucene query for linked files searching * Marge linked files and bib fields results * Searching in background task * Fix search to SQL tests * Localization test * Fix DatabaseSearcherTest * Fix DatabaseSearcherWithBibFilesTest * Fix exportMatches test * Update src/main/java/org/jabref/model/entry/BibEntry.java Co-authored-by: Oliver Kopp * Add SINGLE_ENTRY_LINK to latex field * Remove changelog entries * Remove groups migration from localization * Extract search terms from query (ignore negated terms) * Fix architecture test * Highlight Preview viewer with Postgres regexp_replace * OpenRewrite * Remove onRunning * Set search query listener in the constructor * Fix preview tab scrolling * Use prepared statement to fix escaping * Use prepared statement for sql query * Store the start and end positions for every field * WIP highlight source tab * Fix source tab highlighting * Return regex, case-sensitive flags back to the search bar * Use for search bar flags for unfielded terms * Skip migrations for unfieleded terms * Return regex, case-sensitive CheckBox to search groups dialog * Apply suggestions from code review Co-authored-by: Oliver Kopp * Update JabRef_en.properties * Fix search grammar to support special chars * Create SearchQueryTest.java * Adapt SQL visitor with new grammar * Allow to use quotes without escaping * escape SQL wildcard chars * Reorder methods * Adapt SearchFlagsToExpressionVisitor * Adapt SearchToLuceneVisitor * Adapt SearchQueryExtractorVisitor * Fix tests * Fix DatabaseSearcherTest * Fix search terms patten for highlighting * Highlight source tab field by field according to the search query * Apply suggestions from code review * Update src/main/java/org/jabref/gui/importer/actions/SearchGroupsMigrationAction.java Co-authored-by: Oliver Kopp * Set default operator to AND * remove debug --------- Co-authored-by: Oliver Kopp Co-authored-by: Carl Christian Snethlage <50491877+calixtus@users.noreply.github.com> Co-authored-by: Siedlerchr --- CHANGELOG.md | 4 - build.gradle | 8 + external-libraries.md | 18 +- src/main/antlr4/org/jabref/search/Search.g4 | 97 +- src/main/java/module-info.java | 7 +- src/main/java/org/jabref/Launcher.java | 4 + .../org/jabref/cli/ArgumentProcessor.java | 6 +- src/main/java/org/jabref/gui/JabRefGUI.java | 4 + src/main/java/org/jabref/gui/LibraryTab.java | 37 +- .../java/org/jabref/gui/StateManager.java | 14 +- .../jabref/gui/entryeditor/CommentsTab.java | 8 +- .../gui/entryeditor/DeprecatedFieldsTab.java | 8 +- .../entryeditor/DetailOptionalFieldsTab.java | 8 +- .../jabref/gui/entryeditor/EntryEditor.java | 20 +- .../gui/entryeditor/FieldsEditorTab.java | 12 +- .../ImportantOptionalFieldsTab.java | 8 +- .../entryeditor/OptionalFieldsTabBase.java | 8 +- .../gui/entryeditor/OtherFieldsTab.java | 8 +- .../jabref/gui/entryeditor/PreviewTab.java | 12 +- .../gui/entryeditor/RequiredFieldsTab.java | 8 +- .../org/jabref/gui/entryeditor/SourceTab.java | 117 ++- .../gui/entryeditor/UserDefinedFieldsTab.java | 8 +- .../FulltextSearchResultsTab.java | 6 +- .../gui/exporter/SaveDatabaseAction.java | 8 +- .../ExternalFilesEntryLinker.java | 14 +- .../org/jabref/gui/frame/JabRefFrame.java | 1 - .../org/jabref/gui/groups/GroupDialog.fxml | 12 + .../jabref/gui/groups/GroupDialogView.java | 7 + .../gui/groups/GroupDialogViewModel.java | 30 +- .../jabref/gui/groups/GroupNodeViewModel.java | 32 +- .../java/org/jabref/gui/icon/IconTheme.java | 1 + .../actions/CheckForNewEntryTypesAction.java | 2 +- .../importer/actions/GUIPostOpenAction.java | 6 +- .../actions/MergeReviewIntoCommentAction.java | 2 +- .../importer/actions/OpenDatabaseAction.java | 2 +- .../actions/SearchGroupsMigrationAction.java | 28 +- .../gui/maintable/BibEntryTableViewModel.java | 7 - .../org/jabref/gui/maintable/MainTable.java | 4 +- .../gui/maintable/MainTableColumnFactory.java | 19 - .../gui/maintable/MainTableColumnModel.java | 3 +- .../gui/maintable/MainTableDataModel.java | 46 +- .../gui/preferences/JabRefGuiPreferences.java | 4 +- .../preferences/table/TableTabViewModel.java | 1 - .../org/jabref/gui/preview/PreviewPanel.java | 12 +- .../org/jabref/gui/preview/PreviewViewer.java | 118 +-- .../jabref/gui/search/GlobalSearchBar.java | 80 +- .../RebuildFulltextSearchIndexAction.java | 2 +- .../search/SearchResultsTableDataModel.java | 12 +- .../gui/shared/SharedDatabaseUIManager.java | 10 +- .../jabref/logic/bibtex/BibEntryWriter.java | 33 +- .../org/jabref/logic/bibtex/FieldWriter.java | 15 +- .../org/jabref/logic/exporter/BibWriter.java | 6 + .../preferences/JabRefCliPreferences.java | 8 +- .../jabref/logic/search/DatabaseSearcher.java | 26 +- .../{LuceneManager.java => IndexManager.java} | 90 +- .../jabref/logic/search/PostgreServer.java | 98 ++ .../logic/search/SearchPreferences.java | 25 +- .../search/indexing/BibFieldsIndexer.java | 412 ++++++-- .../indexing/DefaultLinkedFilesIndexer.java | 12 +- .../logic/search/indexing/DocumentReader.java | 10 +- .../query/SearchFlagsToExpressionVisitor.java | 127 +++ .../search/query/SearchQueryConversion.java | 62 ++ .../query/SearchQueryExtractorVisitor.java | 125 +++ .../search/query/SearchToLuceneVisitor.java | 152 +++ .../search/query/SearchToSqlVisitor.java | 552 +++++++++++ .../search/retrieval/BibFieldsSearcher.java | 62 ++ .../logic/search/retrieval/Highlighter.java | 142 +++ .../search/retrieval/LinkedFilesSearcher.java | 131 +++ .../search/retrieval/LuceneSearcher.java | 195 ---- .../org/jabref/logic/shared/DBMSType.java | 2 +- .../shared/event/ConnectionLostEvent.java | 18 +- .../event/SharedEntriesNotPresentEvent.java | 18 +- .../shared/event/UpdateRefusedEvent.java | 34 +- .../PostgresSQLNotificationListener.java | 2 +- .../org/jabref/logic/util/Directories.java | 4 +- .../ConvertLegacyExplicitGroups.java | 3 +- .../migrations/SearchToLuceneMigration.java | 31 - .../migrations/SearchToLuceneVisitor.java | 148 --- .../model/database/BibDatabaseContext.java | 9 + .../java/org/jabref/model/entry/Author.java | 18 +- .../java/org/jabref/model/entry/BibEntry.java | 1 + .../model/entry/field/FieldFactory.java | 2 +- .../org/jabref/model/groups/SearchGroup.java | 24 +- .../search/Analyzer/LatexAwareAnalyzer.java | 26 - .../Analyzer/LatexAwareNGramAnalyzer.java | 32 - .../Analyzer/LatexToUnicodeFoldingFilter.java | 89 -- ...nstants.java => LinkedFilesConstants.java} | 12 +- .../jabref/model/search/PostgreConstants.java | 74 ++ .../search/SearchDisplayMode.java | 2 +- .../org/jabref/model/search/SearchFlags.java | 5 +- .../org/jabref/model/search/SearchQuery.java | 192 ---- .../model/search/query/SearchQuery.java | 95 ++ .../model/search/query/SearchQueryNode.java | 8 + .../search/{ => query}/SearchResult.java | 29 +- .../search/{ => query}/SearchResults.java | 15 +- .../model/search/query/SqlQueryNode.java | 27 + .../java/org/jabref/model/util/Range.java | 4 + src/main/resources/l10n/JabRef_en.properties | 9 +- src/main/resources/tinylog.properties | 6 +- .../org/jabref/cli/ArgumentProcessorTest.java | 4 +- .../gui/entryeditor/CommentsTabTest.java | 4 +- .../logic/search/DatabaseSearcherTest.java | 35 +- .../DatabaseSearcherWithBibFilesTest.java | 66 +- .../logic/search/LuceneQueryParserTest.java | 40 - .../SearchQueryExtractorConversionTest.java | 39 + .../query/SearchQueryFlagsConversionTest.java | 91 ++ .../SearchQueryLuceneConversionTest.java | 66 ++ .../query/SearchQuerySQLConversionTest.java | 888 ++++++++++++++++++ .../logic/search/query/SearchQueryTest.java | 65 ++ .../shared/SynchronizationSimulatorTest.java | 2 +- .../SearchToLuceneMigrationTest.java | 73 -- 111 files changed, 3886 insertions(+), 1602 deletions(-) rename src/main/java/org/jabref/logic/search/{LuceneManager.java => IndexManager.java} (68%) create mode 100644 src/main/java/org/jabref/logic/search/PostgreServer.java create mode 100644 src/main/java/org/jabref/logic/search/query/SearchFlagsToExpressionVisitor.java create mode 100644 src/main/java/org/jabref/logic/search/query/SearchQueryConversion.java create mode 100644 src/main/java/org/jabref/logic/search/query/SearchQueryExtractorVisitor.java create mode 100644 src/main/java/org/jabref/logic/search/query/SearchToLuceneVisitor.java create mode 100644 src/main/java/org/jabref/logic/search/query/SearchToSqlVisitor.java create mode 100644 src/main/java/org/jabref/logic/search/retrieval/BibFieldsSearcher.java create mode 100644 src/main/java/org/jabref/logic/search/retrieval/Highlighter.java create mode 100644 src/main/java/org/jabref/logic/search/retrieval/LinkedFilesSearcher.java delete mode 100644 src/main/java/org/jabref/logic/search/retrieval/LuceneSearcher.java delete mode 100644 src/main/java/org/jabref/migrations/SearchToLuceneMigration.java delete mode 100644 src/main/java/org/jabref/migrations/SearchToLuceneVisitor.java delete mode 100644 src/main/java/org/jabref/model/search/Analyzer/LatexAwareAnalyzer.java delete mode 100644 src/main/java/org/jabref/model/search/Analyzer/LatexAwareNGramAnalyzer.java delete mode 100644 src/main/java/org/jabref/model/search/Analyzer/LatexToUnicodeFoldingFilter.java rename src/main/java/org/jabref/model/search/{SearchFieldConstants.java => LinkedFilesConstants.java} (66%) create mode 100644 src/main/java/org/jabref/model/search/PostgreConstants.java rename src/main/java/org/jabref/{logic => model}/search/SearchDisplayMode.java (63%) delete mode 100644 src/main/java/org/jabref/model/search/SearchQuery.java create mode 100644 src/main/java/org/jabref/model/search/query/SearchQuery.java create mode 100644 src/main/java/org/jabref/model/search/query/SearchQueryNode.java rename src/main/java/org/jabref/model/search/{ => query}/SearchResult.java (72%) rename src/main/java/org/jabref/model/search/{ => query}/SearchResults.java (80%) create mode 100644 src/main/java/org/jabref/model/search/query/SqlQueryNode.java create mode 100644 src/main/java/org/jabref/model/util/Range.java delete mode 100644 src/test/java/org/jabref/logic/search/LuceneQueryParserTest.java create mode 100644 src/test/java/org/jabref/logic/search/query/SearchQueryExtractorConversionTest.java create mode 100644 src/test/java/org/jabref/logic/search/query/SearchQueryFlagsConversionTest.java create mode 100644 src/test/java/org/jabref/logic/search/query/SearchQueryLuceneConversionTest.java create mode 100644 src/test/java/org/jabref/logic/search/query/SearchQuerySQLConversionTest.java create mode 100644 src/test/java/org/jabref/logic/search/query/SearchQueryTest.java delete mode 100644 src/test/java/org/jabref/migrations/SearchToLuceneMigrationTest.java diff --git a/CHANGELOG.md b/CHANGELOG.md index 2f5ba62e7df..b442be867c5 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,7 +12,6 @@ Note that this project **does not** adhere to [Semantic Versioning](https://semv ### Added - We added a "view as BibTeX" option before importing an entry from the citation relation tab. [#11826](https://github.com/JabRef/jabref/issues/11826) -- We added probable search hits instead of exact matches. Sorting by hit score can be done by the new score table column. [#11542](https://github.com/JabRef/jabref/pull/11542) - We added support finding LaTeX-encoded special characters based on plain Unicode and vice versa. [#11542](https://github.com/JabRef/jabref/pull/11542) - When a search hits a file, the file icon of that entry is changed accordingly. [#11542](https://github.com/JabRef/jabref/pull/11542) - We added an AI-based chat for entries with linked PDF files. [#11430](https://github.com/JabRef/jabref/pull/11430) @@ -42,8 +41,6 @@ Note that this project **does not** adhere to [Semantic Versioning](https://semv ### Changed -- The search syntax is changed to [Apache Lucene syntax](https://lucene.apache.org/core/9_11_1/queryparser/org/apache/lucene/queryparser/classic/package-summary.html#Overview) (also to be similar to the [online search syntax](https://docs.jabref.org/collect/import-using-online-bibliographic-database#search-syntax)). [#11542](https://github.com/JabRef/jabref/pull/11542/) -- When searching using a regular expression, one needs to enclose the search string in `/`. [#11542](https://github.com/JabRef/jabref/pull/11542/) - A search in "any" fields ignores the [groups](https://docs.jabref.org/finding-sorting-and-cleaning-entries/groups). [#7996](https://github.com/JabRef/jabref/issues/7996) - When a communication error with an [online service](https://docs.jabref.org/collect/import-using-online-bibliographic-database) occurs, JabRef displays the HTTP error. [#11223](https://github.com/JabRef/jabref/issues/11223) - The Pubmed/Medline Plain importer now imports the PMID field as well [#11488](https://github.com/JabRef/jabref/issues/11488) @@ -103,7 +100,6 @@ Note that this project **does not** adhere to [Semantic Versioning](https://semv ### Removed -- We removed support for case-sensitive and exact search. [#11542](https://github.com/JabRef/jabref/pull/11542) - We removed the description of search strings. [#11542](https://github.com/JabRef/jabref/pull/11542) - We removed support for importing using the SilverPlatterImporter (`Record INSPEC`). [#11576](https://github.com/JabRef/jabref/pull/11576) - We removed support for automatically generating file links using the CLI (`--automaticallySetFileLinks`). diff --git a/build.gradle b/build.gradle index 85e3d12f0d6..116feba772b 100644 --- a/build.gradle +++ b/build.gradle @@ -370,6 +370,9 @@ dependencies { // Even if "compileOnly" is used, IntelliJ always adds to module-info.java. To avoid issues during committing, we use "implementation" instead of "compileOnly" implementation 'io.github.adr:e-adr:2.0.0-SNAPSHOT' + implementation 'io.zonky.test:embedded-postgres:2.0.7' + implementation enforcedPlatform('io.zonky.test.postgres:embedded-postgres-binaries-bom:17.0.0') + testImplementation 'io.github.classgraph:classgraph:4.8.177' testImplementation 'org.junit.jupiter:junit-jupiter:5.11.0' testImplementation 'org.junit.platform:junit-platform-launcher:1.10.3' @@ -793,9 +796,13 @@ jlink { requires 'org.apache.commons.lang3' requires 'org.apache.commons.logging' requires 'org.apache.commons.text' + requires 'org.apache.commons.codec' + requires 'org.apache.commons.io' + requires 'org.apache.commons.compress' requires 'org.freedesktop.dbus' requires 'org.jsoup' requires 'org.slf4j' + requires 'org.tukaani.xz'; uses 'ai.djl.engine.EngineProvider' uses 'ai.djl.repository.RepositoryFactory' uses 'ai.djl.repository.zoo.ZooProvider' @@ -807,6 +814,7 @@ jlink { uses 'org.mariadb.jdbc.authentication.AuthenticationPlugin' uses 'org.mariadb.jdbc.credential.CredentialPlugin' uses 'org.mariadb.jdbc.tls.TlsSocketPlugin' + uses 'org.postgresql.shaded.com.ongres.stringprep.Profile' provides 'org.mariadb.jdbc.tls.TlsSocketPlugin' with 'org.mariadb.jdbc.internal.protocol.tls.DefaultTlsSocketPlugin' provides 'java.sql.Driver' with 'org.postgresql.Driver' diff --git a/external-libraries.md b/external-libraries.md index 61566a72709..257aea187cb 100644 --- a/external-libraries.md +++ b/external-libraries.md @@ -350,10 +350,10 @@ License: MIT ``` ```yaml -Id:io.github.adr:e-adr -Project:EmbeddedArchitecturalDecisionRecords -URL:https://github.com/adr/e-adr/ -License:EPL-2.0 +Id: io.github.adr:e-adr +Project: EmbeddedArchitecturalDecisionRecords +URL: https://github.com/adr/e-adr/ +License: EPL-2.0 ``` ```yaml @@ -363,6 +363,13 @@ URL: https://github.com/java-diff-utils/java-diff-utils License: Apache-2.0 ``` +```yaml +Id: io.zonky.test:embedded-postgres +Project: embedded-postgres +URL: https://github.com/zonkyio/embedded-postgres +License: Apache-2.0 +``` + ```yaml Id: jakarta.annotation:jakarata.annotation-api Project: Jakarta Annotations @@ -798,6 +805,7 @@ de.undercouch:citeproc-java:3.1.0 eu.lestard:doc-annotations:0.2 info.debatty:java-string-similarity:2.0.0 io.github.java-diff-utils:java-diff-utils:4.12 +io.zonky.test:embedded-postgres:2.0.7 jakarta.activation:jakarta.activation-api:2.1.3 jakarta.annotation:jakarta.annotation-api:2.1.1 jakarta.inject:jakarta.inject-api:2.0.1 @@ -813,6 +821,7 @@ net.jodah:typetools:0.6.1 net.synedra:validatorfx:0.5.0 one.jpro.jproutils:tree-showing:0.2.2 org.antlr:antlr4-runtime:4.13.2 +org.apache.commons:commons-compress:1.27.1 org.apache.commons:commons-csv:1.11.0 org.apache.commons:commons-lang3:3.17.0 org.apache.commons:commons-text:1.12.0 @@ -889,6 +898,7 @@ org.slf4j:slf4j-api:2.0.16 org.tinylog:slf4j-tinylog:2.7.0 org.tinylog:tinylog-api:2.7.0 org.tinylog:tinylog-impl:2.7.0 +org.tukaani:xz:1.9 org.yaml:snakeyaml:2.3 pt.davidafsilva.apple:jkeychain:1.1.0 tech.units:indriya:2.2 diff --git a/src/main/antlr4/org/jabref/search/Search.g4 b/src/main/antlr4/org/jabref/search/Search.g4 index e2ee78b6c9d..0a8c28c234c 100644 --- a/src/main/antlr4/org/jabref/search/Search.g4 +++ b/src/main/antlr4/org/jabref/search/Search.g4 @@ -4,49 +4,82 @@ * These search expressions are used for searching the bibtex library. They are heavily used for search groups. */ grammar Search; +options { caseInsensitive = true; } -WS: [ \t] -> skip; // whitespace is ignored/skipped +WS: [ \t\n\r]+ -> skip; // whitespace is ignored/skipped -LPAREN:'('; -RPAREN:')'; +LPAREN: '('; +RPAREN: ')'; -EQUAL:'='; // semantically the same as CONTAINS -EEQUAL:'=='; // semantically the same as MATCHES -NEQUAL:'!='; +EQUAL: '='; // case insensitive contains, semantically the same as CONTAINS +CEQUAL: '=!'; // case sensitive contains -AND:[aA][nN][dD]; // 'and' case insensitive -OR:[oO][rR]; // 'or' case insensitive -CONTAINS:[cC][oO][nN][tT][aA][iI][nN][sS]; // 'contains' case insensitive -MATCHES:[mM][aA][tT][cC][hH][eE][sS]; // 'matches' case insensitive -NOT:[nN][oO][tT]; // 'not' case insensitive +EEQUAL: '=='; // exact match case insensitive, semantically the same as MATCHES +CEEQUAL: '==!'; // exact match case sensitive -STRING:QUOTE (~'"')* QUOTE; -QUOTE:'"'; +REQUAL: '=~'; // regex check case insensitive +CREEQUAL: '=~!'; // regex check case sensitive -FIELDTYPE:LETTER+; -// fragments are not accessible from the code, they are only for describing the grammar better -fragment LETTER : ~[ \t"()=!]; +NEQUAL: '!='; // negated case insensitive contains +NCEQUAL: '!=!'; // negated case sensitive contains +NEEQUAL: '!=='; // negated case insensitive exact match +NCEEQUAL: '!==!'; // negated case sensitive exact match -start: - expression EOF; +NREQUAL: '!=~'; // negated regex check case insensitive +NCREEQUAL: '!=~!'; // negated regex check case sensitive -// labels are used to refer to parts of the rules in the generated code later on -// label=actualThingy -expression: - LPAREN expression RPAREN #parenExpression // example: (author=miller) - | NOT expression #unaryExpression // example: not author = miller - | left=expression operator=AND right=expression #binaryExpression // example: author = miller and title = test - | left=expression operator=OR right=expression #binaryExpression // example: author = miller or title = test - | comparison #atomExpression +AND: 'AND'; +OR: 'OR'; +CONTAINS: 'CONTAINS'; +MATCHES: 'MATCHES'; +NOT: 'NOT'; + +FIELD: [A-Z]+; +STRING_LITERAL: '"' ('\\"' | ~["])* '"'; // " should be escaped with a backslash +TERM: ('\\' [=!~()] | ~[ \t\n\r=!~()])+; // =!~() should be escaped with a backslash + +start + : EOF + | andExpression EOF + ; + +andExpression + : expression+ #implicitAndExpression // example: author = miller year = 2010 --> equivalent to: author = miller AND year = 2010 + ; + +expression + : LPAREN andExpression RPAREN #parenExpression // example: (author = miller) + | NOT expression #negatedExpression // example: NOT author = miller + | left = expression bin_op = AND right = expression #binaryExpression // example: author = miller AND year = 2010 + | left = expression bin_op = OR right = expression #binaryExpression // example: author = miller OR year = 2010 + | comparison #comparisonExpression // example: miller OR author = miller + ; + +comparison + : FIELD operator searchValue // example: author = miller + | searchValue // example: miller ; -comparison: - left=name operator=(CONTAINS | MATCHES | EQUAL | EEQUAL | NEQUAL) right=name // example: author != miller - | right=name // example: miller (search all fields) +operator + : EQUAL + | CEQUAL + | EEQUAL + | CEEQUAL + | REQUAL + | CREEQUAL + | NEQUAL + | NCEQUAL + | NEEQUAL + | NCEEQUAL + | NREQUAL + | NCREEQUAL + | CONTAINS + | MATCHES ; -name: - STRING // example: "miller" - | FIELDTYPE // example: author +searchValue + : STRING_LITERAL + | FIELD + | TERM ; diff --git a/src/main/java/module-info.java b/src/main/java/module-info.java index a199ea29502..49906ad2625 100644 --- a/src/main/java/module-info.java +++ b/src/main/java/module-info.java @@ -97,6 +97,8 @@ // endregion // region: SQL databases + requires embedded.postgres; + requires org.tukaani.xz; requires ojdbc10; requires org.postgresql.jdbc; requires org.mariadb.jdbc; @@ -108,6 +110,7 @@ requires io.github.javadiffutils; requires java.string.similarity; requires org.apache.commons.cli; + requires org.apache.commons.compress; requires org.apache.commons.csv; requires org.apache.commons.io; requires org.apache.commons.lang3; @@ -160,8 +163,8 @@ // endregion // region: Lucene - /** - * In case the version is updated, please also increment {@link org.jabref.model.search.SearchFieldConstants#VERSION} to trigger reindexing. + /* + * In case the version is updated, please also increment {@link org.jabref.model.search.LinkedFilesConstants.VERSION} to trigger reindexing. */ uses org.apache.lucene.codecs.lucene100.Lucene100Codec; requires org.apache.lucene.analysis.common; diff --git a/src/main/java/org/jabref/Launcher.java b/src/main/java/org/jabref/Launcher.java index 8806c7892bf..a60a05bec30 100644 --- a/src/main/java/org/jabref/Launcher.java +++ b/src/main/java/org/jabref/Launcher.java @@ -9,6 +9,7 @@ import org.jabref.gui.util.DefaultFileUpdateMonitor; import org.jabref.logic.UiCommand; import org.jabref.logic.preferences.CliPreferences; +import org.jabref.logic.search.PostgreServer; import org.jabref.logic.util.HeadlessExecutorService; import org.jabref.migrations.PreferencesMigrations; @@ -38,6 +39,9 @@ public static void main(String[] args) { PreferencesMigrations.runMigrations(preferences); + PostgreServer postgreServer = new PostgreServer(); + Injector.setModelOrService(PostgreServer.class, postgreServer); + JabRefGUI.setup(uiCommands, preferences, fileUpdateMonitor); JabRefGUI.launch(JabRefGUI.class, args); } diff --git a/src/main/java/org/jabref/cli/ArgumentProcessor.java b/src/main/java/org/jabref/cli/ArgumentProcessor.java index 1021049647a..3df190f9816 100644 --- a/src/main/java/org/jabref/cli/ArgumentProcessor.java +++ b/src/main/java/org/jabref/cli/ArgumentProcessor.java @@ -53,7 +53,7 @@ import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.BibEntryTypesManager; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; import org.jabref.model.strings.StringUtil; import org.jabref.model.util.DummyFileUpdateMonitor; import org.jabref.model.util.FileUpdateMonitor; @@ -458,8 +458,8 @@ private boolean exportMatches(List loaded) { List matches; try { - // extract current thread task executor from luceneManager - matches = new DatabaseSearcher(query, databaseContext, new CurrentThreadTaskExecutor(), cliPreferences.getFilePreferences()).getMatches(); + // extract current thread task executor from indexManager + matches = new DatabaseSearcher(query, databaseContext, new CurrentThreadTaskExecutor(), cliPreferences).getMatches(); } catch (IOException e) { LOGGER.error("Error occurred when searching", e); return false; diff --git a/src/main/java/org/jabref/gui/JabRefGUI.java b/src/main/java/org/jabref/gui/JabRefGUI.java index 94c223eb6c0..be4054302ff 100644 --- a/src/main/java/org/jabref/gui/JabRefGUI.java +++ b/src/main/java/org/jabref/gui/JabRefGUI.java @@ -32,6 +32,7 @@ import org.jabref.logic.net.ProxyRegisterer; import org.jabref.logic.remote.RemotePreferences; import org.jabref.logic.remote.server.RemoteListenerServerManager; +import org.jabref.logic.search.PostgreServer; import org.jabref.logic.util.BuildInfo; import org.jabref.logic.util.FallbackExceptionHandler; import org.jabref.logic.util.HeadlessExecutorService; @@ -395,6 +396,9 @@ public static void shutdownThreadPools() { LOGGER.trace("Shutting down directoryMonitor"); DirectoryMonitor directoryMonitor = Injector.instantiateModelOrService(DirectoryMonitor.class); directoryMonitor.shutdown(); + LOGGER.trace("Shutting down postgreServer"); + PostgreServer postgreServer = Injector.instantiateModelOrService(PostgreServer.class); + postgreServer.shutdown(); LOGGER.trace("Shutting down HeadlessExecutorService"); HeadlessExecutorService.INSTANCE.shutdownEverything(); LOGGER.trace("Finished shutdownThreadPools"); diff --git a/src/main/java/org/jabref/gui/LibraryTab.java b/src/main/java/org/jabref/gui/LibraryTab.java index 3877db193e0..190ab74b520 100644 --- a/src/main/java/org/jabref/gui/LibraryTab.java +++ b/src/main/java/org/jabref/gui/LibraryTab.java @@ -73,7 +73,7 @@ import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; import org.jabref.logic.pdf.FileAnnotationCache; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.shared.DatabaseLocation; import org.jabref.logic.util.BackgroundTask; import org.jabref.logic.util.TaskExecutor; @@ -93,9 +93,8 @@ import org.jabref.model.entry.event.FieldChangedEvent; 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.groups.GroupTreeNode; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; import org.jabref.model.util.DirectoryMonitor; import org.jabref.model.util.DirectoryMonitorManager; import org.jabref.model.util.FileUpdateMonitor; @@ -168,7 +167,7 @@ private enum PanelMode { MAIN_TABLE, MAIN_TABLE_AND_ENTRY_EDITOR } private final DirectoryMonitorManager directoryMonitorManager; private ImportHandler importHandler; - private LuceneManager luceneManager; + private IndexManager indexManager; private final AiService aiService; @@ -215,7 +214,7 @@ private LibraryTab(BibDatabaseContext bibDatabaseContext, private void initializeComponentsAndListeners(boolean isDummyContext) { if (!isDummyContext) { - createLuceneManager(); + createIndexManager(); } if (tableModel != null) { @@ -226,7 +225,7 @@ private void initializeComponentsAndListeners(boolean isDummyContext) { bibDatabaseContext.getMetaData().registerListener(this); this.selectedGroupsProperty = new SimpleListProperty<>(stateManager.getSelectedGroups(bibDatabaseContext)); - this.tableModel = new MainTableDataModel(getBibDatabaseContext(), preferences, taskExecutor, stateManager, getLuceneManager(), selectedGroupsProperty(), searchQueryProperty(), resultSizeProperty()); + this.tableModel = new MainTableDataModel(getBibDatabaseContext(), preferences, taskExecutor, getIndexManager(), selectedGroupsProperty(), searchQueryProperty(), resultSizeProperty()); new CitationStyleCache(bibDatabaseContext); annotationCache = new FileAnnotationCache(bibDatabaseContext, preferences.getFilePreferences()); @@ -322,17 +321,17 @@ private void onDatabaseLoadingSucceed(ParserResult result) { dataLoadingTask = null; } - public void createLuceneManager() { - luceneManager = new LuceneManager(bibDatabaseContext, taskExecutor, preferences.getFilePreferences()); - stateManager.setLuceneManager(bibDatabaseContext, luceneManager); + public void createIndexManager() { + indexManager = new IndexManager(bibDatabaseContext, taskExecutor, preferences); + stateManager.setIndexManager(bibDatabaseContext, indexManager); } - public LuceneManager getLuceneManager() { - return luceneManager; + public IndexManager getIndexManager() { + return indexManager; } - public void closeLuceneManger() { - luceneManager.close(); + public void closeIndexManger() { + indexManager.close(); } private void onDatabaseLoadingFailed(Exception ex) { @@ -794,11 +793,11 @@ private void onClosed(Event event) { LOGGER.error("Problem when closing directory monitor", e); } try { - if (luceneManager != null) { - luceneManager.close(); + if (indexManager != null) { + indexManager.close(); } } catch (RuntimeException e) { - LOGGER.error("Problem when closing lucene indexer", e); + LOGGER.error("Problem when closing index manager", e); } try { AutosaveManager.shutdown(bibDatabaseContext); @@ -1151,17 +1150,17 @@ private class IndexUpdateListener { @Subscribe public void listen(EntriesAddedEvent addedEntryEvent) { - luceneManager.addToIndex(addedEntryEvent.getBibEntries()); + indexManager.addToIndex(addedEntryEvent.getBibEntries()); } @Subscribe public void listen(EntriesRemovedEvent removedEntriesEvent) { - luceneManager.removeFromIndex(removedEntriesEvent.getBibEntries()); + indexManager.removeFromIndex(removedEntriesEvent.getBibEntries()); } @Subscribe public void listen(FieldChangedEvent fieldChangedEvent) { - luceneManager.updateEntry(fieldChangedEvent.getBibEntry(), fieldChangedEvent.getOldValue(), fieldChangedEvent.getNewValue(), fieldChangedEvent.getField().equals(StandardField.FILE)); + indexManager.updateEntry(fieldChangedEvent); } } diff --git a/src/main/java/org/jabref/gui/StateManager.java b/src/main/java/org/jabref/gui/StateManager.java index e75ffcdc89e..435c018619d 100644 --- a/src/main/java/org/jabref/gui/StateManager.java +++ b/src/main/java/org/jabref/gui/StateManager.java @@ -27,12 +27,12 @@ import org.jabref.gui.util.CustomLocalDragboard; import org.jabref.gui.util.DialogWindowState; import org.jabref.gui.util.OptionalObjectProperty; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.BackgroundTask; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; import org.jabref.model.groups.GroupTreeNode; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; import com.tobiasdiez.easybind.EasyBind; import com.tobiasdiez.easybind.EasyBinding; @@ -61,7 +61,7 @@ public class StateManager { private final OptionalObjectProperty activeTab = OptionalObjectProperty.empty(); private final ObservableList selectedEntries = FXCollections.observableArrayList(); private final ObservableMap> selectedGroups = FXCollections.observableHashMap(); - private final ObservableMap luceneManagers = FXCollections.observableHashMap(); + private final ObservableMap indexManagers = FXCollections.observableHashMap(); private final OptionalObjectProperty activeSearchQuery = OptionalObjectProperty.empty(); private final OptionalObjectProperty activeGlobalSearchQuery = OptionalObjectProperty.empty(); private final IntegerProperty searchResultSize = new SimpleIntegerProperty(0); @@ -128,12 +128,12 @@ public void clearSelectedGroups(BibDatabaseContext context) { selectedGroups.computeIfAbsent(context.getUid(), k -> FXCollections.observableArrayList()).clear(); } - public void setLuceneManager(BibDatabaseContext database, LuceneManager luceneManager) { - luceneManagers.put(database.getUid(), luceneManager); + public void setIndexManager(BibDatabaseContext database, IndexManager indexManager) { + indexManagers.put(database.getUid(), indexManager); } - public Optional getLuceneManager(BibDatabaseContext database) { - return Optional.ofNullable(luceneManagers.get(database.getUid())); + public Optional getIndexManager(BibDatabaseContext database) { + return Optional.ofNullable(indexManagers.get(database.getUid())); } public Optional getActiveDatabase() { diff --git a/src/main/java/org/jabref/gui/entryeditor/CommentsTab.java b/src/main/java/org/jabref/gui/entryeditor/CommentsTab.java index 95e2b927dd9..a44d7ef4956 100644 --- a/src/main/java/org/jabref/gui/entryeditor/CommentsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/CommentsTab.java @@ -29,14 +29,14 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; 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.StandardField; import org.jabref.model.entry.field.UserSpecificCommentField; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class CommentsTab extends FieldsEditorTab { public static final String NAME = "Comments"; @@ -56,7 +56,7 @@ public CommentsTab(GuiPreferences preferences, ThemeManager themeManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { super( false, @@ -70,7 +70,7 @@ public CommentsTab(GuiPreferences preferences, themeManager, taskExecutor, journalAbbreviationRepository, - luceneManager, + indexManager, searchQueryProperty ); this.defaultOwner = preferences.getOwnerPreferences().getDefaultOwner().toLowerCase(Locale.ROOT).replaceAll("[^a-z0-9]", "-"); diff --git a/src/main/java/org/jabref/gui/entryeditor/DeprecatedFieldsTab.java b/src/main/java/org/jabref/gui/entryeditor/DeprecatedFieldsTab.java index 45248e89dfe..4fda9b83cc0 100644 --- a/src/main/java/org/jabref/gui/entryeditor/DeprecatedFieldsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/DeprecatedFieldsTab.java @@ -19,7 +19,7 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.database.BibDatabaseMode; @@ -27,7 +27,7 @@ import org.jabref.model.entry.BibEntryType; import org.jabref.model.entry.BibEntryTypesManager; import org.jabref.model.entry.field.Field; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; import com.tobiasdiez.easybind.EasyBind; @@ -47,9 +47,9 @@ public DeprecatedFieldsTab(BibDatabaseContext databaseContext, BibEntryTypesManager entryTypesManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { - super(false, databaseContext, suggestionProviders, undoManager, undoAction, redoAction, dialogService, preferences, themeManager, taskExecutor, journalAbbreviationRepository, luceneManager, searchQueryProperty); + super(false, databaseContext, suggestionProviders, undoManager, undoAction, redoAction, dialogService, preferences, themeManager, taskExecutor, journalAbbreviationRepository, indexManager, searchQueryProperty); this.entryTypesManager = entryTypesManager; setText(Localization.lang("Deprecated fields")); diff --git a/src/main/java/org/jabref/gui/entryeditor/DetailOptionalFieldsTab.java b/src/main/java/org/jabref/gui/entryeditor/DetailOptionalFieldsTab.java index 4458d80a405..fb03e121a52 100644 --- a/src/main/java/org/jabref/gui/entryeditor/DetailOptionalFieldsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/DetailOptionalFieldsTab.java @@ -11,11 +11,11 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntryTypesManager; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class DetailOptionalFieldsTab extends OptionalFieldsTabBase { @@ -32,7 +32,7 @@ public DetailOptionalFieldsTab(BibDatabaseContext databaseContext, BibEntryTypesManager entryTypesManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { super( Localization.lang("Optional fields 2"), @@ -48,7 +48,7 @@ public DetailOptionalFieldsTab(BibDatabaseContext databaseContext, entryTypesManager, taskExecutor, journalAbbreviationRepository, - luceneManager, + indexManager, searchQueryProperty ); } diff --git a/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java b/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java index 6a382a7a59e..24942579e64 100644 --- a/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java +++ b/src/main/java/org/jabref/gui/entryeditor/EntryEditor.java @@ -175,11 +175,11 @@ private void setupDragAndDrop(LibraryTab libraryTab) { switch (event.getTransferMode()) { case COPY -> { LOGGER.debug("Mode COPY"); - fileLinker.copyFilesToFileDirAndAddToEntry(entry, draggedFiles, libraryTab.getLuceneManager()); + fileLinker.copyFilesToFileDirAndAddToEntry(entry, draggedFiles, libraryTab.getIndexManager()); } case MOVE -> { LOGGER.debug("Mode MOVE"); - fileLinker.moveFilesToFileDirRenameAndAddToEntry(entry, draggedFiles, libraryTab.getLuceneManager()); + fileLinker.moveFilesToFileDirRenameAndAddToEntry(entry, draggedFiles, libraryTab.getIndexManager()); } case LINK -> { LOGGER.debug("Mode LINK"); @@ -276,21 +276,21 @@ private void navigateToNextEntry() { private List createTabs() { List tabs = new LinkedList<>(); - tabs.add(new PreviewTab(databaseContext, dialogService, preferences, themeManager, taskExecutor, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); + tabs.add(new PreviewTab(databaseContext, dialogService, preferences, themeManager, taskExecutor, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); // Required, optional (important+detail), deprecated, and "other" fields - tabs.add(new RequiredFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); - tabs.add(new ImportantOptionalFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); - tabs.add(new DetailOptionalFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); - tabs.add(new DeprecatedFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); - tabs.add(new OtherFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); + tabs.add(new RequiredFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); + tabs.add(new ImportantOptionalFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); + tabs.add(new DetailOptionalFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); + tabs.add(new DeprecatedFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); + tabs.add(new OtherFieldsTab(databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, bibEntryTypesManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); // Comment Tab: Tab for general and user-specific comments - tabs.add(new CommentsTab(preferences, databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, themeManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); + tabs.add(new CommentsTab(preferences, databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, themeManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); Map> entryEditorTabList = getAdditionalUserConfiguredTabs(); for (Map.Entry> tab : entryEditorTabList.entrySet()) { - tabs.add(new UserDefinedFieldsTab(tab.getKey(), tab.getValue(), databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, taskExecutor, journalAbbreviationRepository, libraryTab.getLuceneManager(), libraryTab.searchQueryProperty())); + tabs.add(new UserDefinedFieldsTab(tab.getKey(), tab.getValue(), databaseContext, libraryTab.getSuggestionProviders(), undoManager, undoAction, redoAction, dialogService, preferences, themeManager, taskExecutor, journalAbbreviationRepository, libraryTab.getIndexManager(), libraryTab.searchQueryProperty())); } tabs.add(new MathSciNetTab()); diff --git a/src/main/java/org/jabref/gui/entryeditor/FieldsEditorTab.java b/src/main/java/org/jabref/gui/entryeditor/FieldsEditorTab.java index 41b29ee8f42..b9506739495 100644 --- a/src/main/java/org/jabref/gui/entryeditor/FieldsEditorTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/FieldsEditorTab.java @@ -36,12 +36,12 @@ import org.jabref.gui.undo.UndoAction; import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.field.Field; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; import com.tobiasdiez.easybind.EasyBind; import com.tobiasdiez.easybind.Subscription; @@ -64,7 +64,7 @@ abstract class FieldsEditorTab extends EntryEditorTab implements OffersPreview { private final JournalAbbreviationRepository journalAbbreviationRepository; private PreviewPanel previewPanel; private final UndoManager undoManager; - private final LuceneManager luceneManager; + private final IndexManager indexManager; private final OptionalObjectProperty searchQueryProperty; private Collection fields = new ArrayList<>(); @SuppressWarnings("FieldCanBeLocal") @@ -81,7 +81,7 @@ public FieldsEditorTab(boolean compressed, ThemeManager themeManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { this.isCompressed = compressed; this.databaseContext = Objects.requireNonNull(databaseContext); @@ -94,7 +94,7 @@ public FieldsEditorTab(boolean compressed, this.themeManager = themeManager; this.taskExecutor = Objects.requireNonNull(taskExecutor); this.journalAbbreviationRepository = Objects.requireNonNull(journalAbbreviationRepository); - this.luceneManager = luceneManager; + this.indexManager = indexManager; this.searchQueryProperty = searchQueryProperty; } @@ -265,7 +265,7 @@ private void initPanel() { preferences, themeManager, taskExecutor, - luceneManager, + indexManager, searchQueryProperty); EasyBind.subscribe(preferences.getPreviewPreferences().showPreviewAsExtraTabProperty(), show -> { if (show) { diff --git a/src/main/java/org/jabref/gui/entryeditor/ImportantOptionalFieldsTab.java b/src/main/java/org/jabref/gui/entryeditor/ImportantOptionalFieldsTab.java index 64004b70848..70d831742c3 100644 --- a/src/main/java/org/jabref/gui/entryeditor/ImportantOptionalFieldsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/ImportantOptionalFieldsTab.java @@ -11,11 +11,11 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntryTypesManager; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class ImportantOptionalFieldsTab extends OptionalFieldsTabBase { @@ -32,7 +32,7 @@ public ImportantOptionalFieldsTab(BibDatabaseContext databaseContext, BibEntryTypesManager entryTypesManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { super( Localization.lang("Optional fields"), @@ -48,7 +48,7 @@ public ImportantOptionalFieldsTab(BibDatabaseContext databaseContext, entryTypesManager, taskExecutor, journalAbbreviationRepository, - luceneManager, + indexManager, searchQueryProperty ); } diff --git a/src/main/java/org/jabref/gui/entryeditor/OptionalFieldsTabBase.java b/src/main/java/org/jabref/gui/entryeditor/OptionalFieldsTabBase.java index 0d59a31bbf9..a1d62df4303 100644 --- a/src/main/java/org/jabref/gui/entryeditor/OptionalFieldsTabBase.java +++ b/src/main/java/org/jabref/gui/entryeditor/OptionalFieldsTabBase.java @@ -18,7 +18,7 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.database.BibDatabaseMode; @@ -26,7 +26,7 @@ import org.jabref.model.entry.BibEntryType; import org.jabref.model.entry.BibEntryTypesManager; import org.jabref.model.entry.field.Field; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class OptionalFieldsTabBase extends FieldsEditorTab { private final BibEntryTypesManager entryTypesManager; @@ -45,7 +45,7 @@ public OptionalFieldsTabBase(String title, BibEntryTypesManager entryTypesManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { super(true, databaseContext, @@ -58,7 +58,7 @@ public OptionalFieldsTabBase(String title, themeManager, taskExecutor, journalAbbreviationRepository, - luceneManager, + indexManager, searchQueryProperty); this.entryTypesManager = entryTypesManager; this.isImportantOptionalFields = isImportantOptionalFields; diff --git a/src/main/java/org/jabref/gui/entryeditor/OtherFieldsTab.java b/src/main/java/org/jabref/gui/entryeditor/OtherFieldsTab.java index 5fda9368885..2e9aed08484 100644 --- a/src/main/java/org/jabref/gui/entryeditor/OtherFieldsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/OtherFieldsTab.java @@ -22,7 +22,7 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.database.BibDatabaseMode; @@ -33,7 +33,7 @@ import org.jabref.model.entry.field.InternalField; import org.jabref.model.entry.field.StandardField; import org.jabref.model.entry.field.UserSpecificCommentField; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class OtherFieldsTab extends FieldsEditorTab { @@ -52,7 +52,7 @@ public OtherFieldsTab(BibDatabaseContext databaseContext, BibEntryTypesManager entryTypesManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { super(false, databaseContext, @@ -65,7 +65,7 @@ public OtherFieldsTab(BibDatabaseContext databaseContext, themeManager, taskExecutor, journalAbbreviationRepository, - luceneManager, + indexManager, searchQueryProperty); this.entryTypesManager = entryTypesManager; diff --git a/src/main/java/org/jabref/gui/entryeditor/PreviewTab.java b/src/main/java/org/jabref/gui/entryeditor/PreviewTab.java index 52b51b67b7e..b2ba9839749 100644 --- a/src/main/java/org/jabref/gui/entryeditor/PreviewTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/PreviewTab.java @@ -7,11 +7,11 @@ import org.jabref.gui.theme.ThemeManager; import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class PreviewTab extends EntryEditorTab implements OffersPreview { public static final String NAME = "Preview"; @@ -20,7 +20,7 @@ public class PreviewTab extends EntryEditorTab implements OffersPreview { private final GuiPreferences preferences; private final ThemeManager themeManager; private final TaskExecutor taskExecutor; - private final LuceneManager luceneManager; + private final IndexManager indexManager; private final OptionalObjectProperty searchQueryProperty; private PreviewPanel previewPanel; @@ -29,14 +29,14 @@ public PreviewTab(BibDatabaseContext databaseContext, GuiPreferences preferences, ThemeManager themeManager, TaskExecutor taskExecutor, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { this.databaseContext = databaseContext; this.dialogService = dialogService; this.preferences = preferences; this.themeManager = themeManager; this.taskExecutor = taskExecutor; - this.luceneManager = luceneManager; + this.indexManager = indexManager; this.searchQueryProperty = searchQueryProperty; setGraphic(IconTheme.JabRefIcons.TOGGLE_ENTRY_PREVIEW.getGraphicNode()); @@ -65,7 +65,7 @@ public boolean shouldShow(BibEntry entry) { @Override protected void bindToEntry(BibEntry entry) { if (previewPanel == null) { - previewPanel = new PreviewPanel(databaseContext, dialogService, preferences.getKeyBindingRepository(), preferences, themeManager, taskExecutor, luceneManager, searchQueryProperty); + previewPanel = new PreviewPanel(databaseContext, dialogService, preferences.getKeyBindingRepository(), preferences, themeManager, taskExecutor, indexManager, searchQueryProperty); setContent(previewPanel); } diff --git a/src/main/java/org/jabref/gui/entryeditor/RequiredFieldsTab.java b/src/main/java/org/jabref/gui/entryeditor/RequiredFieldsTab.java index 5966c329fda..78ed8ffbe8f 100644 --- a/src/main/java/org/jabref/gui/entryeditor/RequiredFieldsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/RequiredFieldsTab.java @@ -18,7 +18,7 @@ import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; @@ -27,7 +27,7 @@ import org.jabref.model.entry.field.Field; import org.jabref.model.entry.field.InternalField; import org.jabref.model.entry.field.OrFields; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class RequiredFieldsTab extends FieldsEditorTab { @@ -45,10 +45,10 @@ public RequiredFieldsTab(BibDatabaseContext databaseContext, BibEntryTypesManager entryTypesManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { super(false, databaseContext, suggestionProviders, undoManager, undoAction, redoAction, dialogService, - preferences, themeManager, taskExecutor, journalAbbreviationRepository, luceneManager, searchQueryProperty); + preferences, themeManager, taskExecutor, journalAbbreviationRepository, indexManager, searchQueryProperty); this.entryTypesManager = entryTypesManager; setText(Localization.lang("Required fields")); setTooltip(new Tooltip(Localization.lang("Show required fields"))); diff --git a/src/main/java/org/jabref/gui/entryeditor/SourceTab.java b/src/main/java/org/jabref/gui/entryeditor/SourceTab.java index 26cf33bb55a..629a991e0e8 100644 --- a/src/main/java/org/jabref/gui/entryeditor/SourceTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/SourceTab.java @@ -3,11 +3,10 @@ import java.io.IOException; import java.io.StringReader; import java.io.StringWriter; +import java.util.List; import java.util.Map; import java.util.Objects; import java.util.Optional; -import java.util.regex.Matcher; -import java.util.regex.Pattern; import javax.swing.undo.UndoManager; @@ -42,15 +41,16 @@ import org.jabref.logic.importer.ParserResult; import org.jabref.logic.importer.fileformat.BibtexParser; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.os.OS; +import org.jabref.logic.search.retrieval.Highlighter; import org.jabref.model.database.BibDatabase; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.database.BibDatabaseMode; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.BibEntryTypesManager; import org.jabref.model.entry.field.Field; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; import org.jabref.model.util.FileUpdateMonitor; +import org.jabref.model.util.Range; import de.saxsys.mvvmfx.utils.validation.ObservableRuleBasedValidator; import de.saxsys.mvvmfx.utils.validation.ValidationMessage; @@ -63,6 +63,8 @@ public class SourceTab extends EntryEditorTab { private static final Logger LOGGER = LoggerFactory.getLogger(SourceTab.class); + private static final String TEXT_STYLE = "text"; + private static final String SEARCH_STYLE = "search"; private final FieldPreferences fieldPreferences; private final BibDatabaseMode mode; private final UndoManager undoManager; @@ -73,30 +75,11 @@ public class SourceTab extends EntryEditorTab { private final DialogService dialogService; private final BibEntryTypesManager entryTypesManager; private final KeyBindingRepository keyBindingRepository; - private Optional searchHighlightPattern = Optional.empty(); + private final OptionalObjectProperty searchQueryProperty; + private Map fieldPositions; private CodeArea codeArea; private BibEntry previousEntry; - private class EditAction extends SimpleCommand { - - private final StandardActions command; - - public EditAction(StandardActions command) { - this.command = command; - } - - @Override - public void execute() { - switch (command) { - case COPY -> codeArea.copy(); - case CUT -> codeArea.cut(); - case PASTE -> codeArea.paste(); - case SELECT_ALL -> codeArea.selectAll(); - } - codeArea.requestFocus(); - } - } - public SourceTab(BibDatabaseContext bibDatabaseContext, CountingUndoManager undoManager, FieldPreferences fieldPreferences, @@ -117,33 +100,57 @@ public SourceTab(BibDatabaseContext bibDatabaseContext, this.dialogService = dialogService; this.entryTypesManager = entryTypesManager; this.keyBindingRepository = keyBindingRepository; - - searchQueryProperty.addListener((observable, oldValue, newValue) -> { - searchHighlightPattern = newValue.flatMap(SearchQuery::getPatternForWords); - highlightSearchPattern(); - }); + this.searchQueryProperty = searchQueryProperty; + searchQueryProperty.addListener((observable, oldValue, newValue) -> highlightSearchPattern()); } private void highlightSearchPattern() { - if (codeArea != null) { - codeArea.setStyleClass(0, codeArea.getLength(), "text"); - if (searchHighlightPattern.isPresent()) { - Matcher matcher = searchHighlightPattern.get().matcher(codeArea.getText()); - while (matcher.find()) { - for (int i = 0; i <= matcher.groupCount(); i++) { - codeArea.setStyleClass(matcher.start(), matcher.end(), "search"); - } - } + if (codeArea == null || searchQueryProperty.get().isEmpty()) { + return; + } + + codeArea.setStyleClass(0, codeArea.getLength(), TEXT_STYLE); + Map, List> searchTermsMap = Highlighter.groupTermsByField(searchQueryProperty.get().get()); + + searchTermsMap.forEach((optionalField, terms) -> { + Optional searchPattern = Highlighter.buildSearchPattern(terms); + if (searchPattern.isEmpty()) { + return; + } + + if (optionalField.isPresent()) { + highlightField(optionalField.get(), searchPattern.get()); + } else { + fieldPositions.keySet().forEach(field -> highlightField(field, searchPattern.get())); } + }); + } + + private void highlightField(Field field, String searchPattern) { + Range fieldPosition = fieldPositions.get(field); + if (fieldPosition == null) { + return; + } + + int start = fieldPosition.start(); + int end = fieldPosition.end(); + List matchedPositions = Highlighter.findMatchPositions(codeArea.getText(start, end), searchPattern); + + for (Range range : matchedPositions) { + codeArea.setStyleClass(start + range.start() - 1, start + range.end(), SEARCH_STYLE); } } private String getSourceString(BibEntry entry, BibDatabaseMode type, FieldPreferences fieldPreferences) throws IOException { StringWriter writer = new StringWriter(); - BibWriter bibWriter = new BibWriter(writer, OS.NEWLINE); + BibWriter bibWriter = new BibWriter(writer, "\n"); // JavaFX works with LF only FieldWriter fieldWriter = FieldWriter.buildIgnoreHashes(fieldPreferences); - new BibEntryWriter(fieldWriter, entryTypesManager).write(entry, bibWriter, type); - return writer.toString(); + BibEntryWriter bibEntryWriter = new BibEntryWriter(fieldWriter, entryTypesManager); + bibEntryWriter.write(entry, bibWriter, type, true); + fieldPositions = bibEntryWriter.getFieldPositions(); + String sourceString = writer.toString(); + writer.close(); + return sourceString; } /* Work around for different input methods. @@ -338,12 +345,30 @@ private void storeSource(BibEntry outOfFocusEntry, String text) { private void listenForSaveKeybinding(KeyEvent event) { keyBindingRepository.mapToKeyBinding(event).ifPresent(binding -> { - switch (binding) { - case SAVE_DATABASE, SAVE_ALL, SAVE_DATABASE_AS -> { - storeSource(currentEntry, codeArea.textProperty().getValue()); - } + case SAVE_DATABASE, SAVE_ALL, SAVE_DATABASE_AS -> + storeSource(currentEntry, codeArea.textProperty().getValue()); } }); } + + private class EditAction extends SimpleCommand { + + private final StandardActions command; + + public EditAction(StandardActions command) { + this.command = command; + } + + @Override + public void execute() { + switch (command) { + case COPY -> codeArea.copy(); + case CUT -> codeArea.cut(); + case PASTE -> codeArea.paste(); + case SELECT_ALL -> codeArea.selectAll(); + } + codeArea.requestFocus(); + } + } } diff --git a/src/main/java/org/jabref/gui/entryeditor/UserDefinedFieldsTab.java b/src/main/java/org/jabref/gui/entryeditor/UserDefinedFieldsTab.java index 3e09cdb59e8..c1aedc08f3e 100644 --- a/src/main/java/org/jabref/gui/entryeditor/UserDefinedFieldsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/UserDefinedFieldsTab.java @@ -15,12 +15,12 @@ import org.jabref.gui.undo.UndoAction; import org.jabref.gui.util.OptionalObjectProperty; import org.jabref.logic.journals.JournalAbbreviationRepository; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.TaskExecutor; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.field.Field; -import org.jabref.model.search.SearchQuery; +import org.jabref.model.search.query.SearchQuery; public class UserDefinedFieldsTab extends FieldsEditorTab { private final LinkedHashSet fields; @@ -37,9 +37,9 @@ public UserDefinedFieldsTab(String name, ThemeManager themeManager, TaskExecutor taskExecutor, JournalAbbreviationRepository journalAbbreviationRepository, - LuceneManager luceneManager, + IndexManager indexManager, OptionalObjectProperty searchQueryProperty) { - super(false, databaseContext, suggestionProviders, undoManager, undoAction, redoAction, dialogService, preferences, themeManager, taskExecutor, journalAbbreviationRepository, luceneManager, searchQueryProperty); + super(false, databaseContext, suggestionProviders, undoManager, undoAction, redoAction, dialogService, preferences, themeManager, taskExecutor, journalAbbreviationRepository, indexManager, searchQueryProperty); this.fields = new LinkedHashSet<>(fields); diff --git a/src/main/java/org/jabref/gui/entryeditor/fileannotationtab/FulltextSearchResultsTab.java b/src/main/java/org/jabref/gui/entryeditor/fileannotationtab/FulltextSearchResultsTab.java index 01441568e60..5e98f7d737d 100644 --- a/src/main/java/org/jabref/gui/entryeditor/fileannotationtab/FulltextSearchResultsTab.java +++ b/src/main/java/org/jabref/gui/entryeditor/fileannotationtab/FulltextSearchResultsTab.java @@ -34,9 +34,9 @@ import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.LinkedFile; import org.jabref.model.search.SearchFlags; -import org.jabref.model.search.SearchQuery; -import org.jabref.model.search.SearchResult; -import org.jabref.model.search.SearchResults; +import org.jabref.model.search.query.SearchQuery; +import org.jabref.model.search.query.SearchResult; +import org.jabref.model.search.query.SearchResults; import org.slf4j.Logger; import org.slf4j.LoggerFactory; diff --git a/src/main/java/org/jabref/gui/exporter/SaveDatabaseAction.java b/src/main/java/org/jabref/gui/exporter/SaveDatabaseAction.java index 6a2dd9cb42b..10487ead6ad 100644 --- a/src/main/java/org/jabref/gui/exporter/SaveDatabaseAction.java +++ b/src/main/java/org/jabref/gui/exporter/SaveDatabaseAction.java @@ -139,10 +139,10 @@ boolean saveAs(Path file, SaveDatabaseMode mode) { Optional databasePath = context.getDatabasePath(); if (databasePath.isPresent()) { - // Close AutosaveManager, BackupManager, and LuceneManager for original library + // Close AutosaveManager, BackupManager, and IndexManager for original library AutosaveManager.shutdown(context); BackupManager.shutdown(context, this.preferences.getFilePreferences().getBackupDirectory(), preferences.getFilePreferences().shouldCreateBackup()); - libraryTab.closeLuceneManger(); + libraryTab.closeIndexManger(); } // Set new location @@ -160,10 +160,10 @@ boolean saveAs(Path file, SaveDatabaseMode mode) { context.setDatabasePath(file); libraryTab.updateTabTitle(false); - // Reset (here: uninstall and install again) AutosaveManager, BackupManager and LuceneManager for the new file name + // Reset (here: uninstall and install again) AutosaveManager, BackupManager and IndexManager for the new file name libraryTab.resetChangeMonitor(); libraryTab.installAutosaveManagerAndBackupManager(); - libraryTab.createLuceneManager(); + libraryTab.createIndexManager(); preferences.getLastFilesOpenedPreferences().getFileHistory().newFile(file); } diff --git a/src/main/java/org/jabref/gui/externalfiles/ExternalFilesEntryLinker.java b/src/main/java/org/jabref/gui/externalfiles/ExternalFilesEntryLinker.java index 85b1a83213e..fdfd00dfb75 100644 --- a/src/main/java/org/jabref/gui/externalfiles/ExternalFilesEntryLinker.java +++ b/src/main/java/org/jabref/gui/externalfiles/ExternalFilesEntryLinker.java @@ -17,7 +17,7 @@ import org.jabref.logic.cleanup.MoveFilesCleanup; import org.jabref.logic.cleanup.RenamePdfCleanup; import org.jabref.logic.l10n.Localization; -import org.jabref.logic.search.LuceneManager; +import org.jabref.logic.search.IndexManager; import org.jabref.logic.util.io.FileNameCleaner; import org.jabref.logic.util.io.FileUtil; import org.jabref.model.database.BibDatabaseContext; @@ -79,19 +79,19 @@ public void addFilesToEntry(BibEntry entry, List files) { } } - public void moveFilesToFileDirRenameAndAddToEntry(BibEntry entry, List files, LuceneManager luceneManager) { - try (AutoCloseable blocker = luceneManager.blockLinkedFileIndexer()) { + public void moveFilesToFileDirRenameAndAddToEntry(BibEntry entry, List files, IndexManager indexManager) { + try (AutoCloseable blocker = indexManager.blockLinkedFileIndexer()) { addFilesToEntry(entry, files); moveLinkedFilesToFileDir(entry); renameLinkedFilesToPattern(entry); } catch (Exception e) { LOGGER.error("Could not block LinkedFilesIndexer", e); } - luceneManager.updateAfterDropFiles(entry); + indexManager.updateAfterDropFiles(entry); } - public void copyFilesToFileDirAndAddToEntry(BibEntry entry, List files, LuceneManager luceneManager) { - try (AutoCloseable blocker = luceneManager.blockLinkedFileIndexer()) { + public void copyFilesToFileDirAndAddToEntry(BibEntry entry, List files, IndexManager indexManager) { + try (AutoCloseable blocker = indexManager.blockLinkedFileIndexer()) { for (Path file : files) { copyFileToFileDir(file) .ifPresent(copiedFile -> addFilesToEntry(entry, Collections.singletonList(copiedFile))); @@ -100,7 +100,7 @@ public void copyFilesToFileDirAndAddToEntry(BibEntry entry, List files, Lu } catch (Exception e) { LOGGER.error("Could not block LinkedFilesIndexer", e); } - luceneManager.updateAfterDropFiles(entry); + indexManager.updateAfterDropFiles(entry); } private List getValidFileNames(List filesToAdd) { diff --git a/src/main/java/org/jabref/gui/frame/JabRefFrame.java b/src/main/java/org/jabref/gui/frame/JabRefFrame.java index c40a946cffc..2b7ffb38497 100644 --- a/src/main/java/org/jabref/gui/frame/JabRefFrame.java +++ b/src/main/java/org/jabref/gui/frame/JabRefFrame.java @@ -524,7 +524,6 @@ private OpenDatabaseAction getOpenDatabaseAction() { * Refreshes the ui after preferences changes */ public void refresh() { - globalSearchBar.updateHintVisibility(); getLibraryTabs().forEach(LibraryTab::setupMainPanel); getLibraryTabs().forEach(tab -> tab.getMainTable().getTableModel().resetFieldFormatter()); } diff --git a/src/main/java/org/jabref/gui/groups/GroupDialog.fxml b/src/main/java/org/jabref/gui/groups/GroupDialog.fxml index 9ec2a307ad1..b1375a4776f 100644 --- a/src/main/java/org/jabref/gui/groups/GroupDialog.fxml +++ b/src/main/java/org/jabref/gui/groups/GroupDialog.fxml @@ -121,6 +121,18 @@