diff --git a/CHANGELOG.md b/CHANGELOG.md index a67ee4884d6..9e3270eb674 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -59,6 +59,7 @@ Note that this project **does not** adhere to [Semantic Versioning](http://semve - We moved the custom entry types dialog into the preferences dialog. [#9760](https://github.com/JabRef/jabref/pull/9760) - We moved the manage content selectors dialog to the library properties. [#9768](https://github.com/JabRef/jabref/pull/9768) - We moved the preferences menu command from the options menu to the file menu. [#9768](https://github.com/JabRef/jabref/pull/9768) +- We reworked the cross ref labels in the entry editor and added a right click menu. [#10046](https://github.com/JabRef/jabref/pull/10046) - We reorganized the order of tabs and settings in the library properties. [#9836](https://github.com/JabRef/jabref/pull/9836) - We changed the handling of an "overflow" of authors at `[authIniN]`: JabRef uses `+` to indicate an overflow. Example: `[authIni2]` produces `A+` (instead of `AB`) for `Aachen and Berlin and Chemnitz`. [#9703](https://github.com/JabRef/jabref/pull/9703) - We moved the preferences option to open the last edited files on startup to the 'General' tab. [#9808](https://github.com/JabRef/jabref/pull/9808) diff --git a/build.gradle b/build.gradle index 7a75a7847fd..73787f662a8 100644 --- a/build.gradle +++ b/build.gradle @@ -88,6 +88,7 @@ repositories { maven { url 'https://oss.sonatype.org/content/groups/public' } maven { url 'https://s01.oss.sonatype.org/content/repositories/snapshots/' } maven { url 'https://jitpack.io' } + maven { url 'https://sandec.jfrog.io/artifactory/repo' } } configurations { @@ -176,7 +177,11 @@ dependencies { implementation 'com.tobiasdiez:easybind:2.2.1-SNAPSHOT' implementation 'org.fxmisc.flowless:flowless:0.7.0' implementation 'org.fxmisc.richtext:richtextfx:0.11.0' - implementation 'com.jfoenix:jfoenix:9.0.10' + implementation (group: 'com.dlsc.gemsfx', name: 'gemsfx', version: '1.74.0') { + exclude module: 'javax.inject' // Split package, use only jakarta.inject + exclude group: 'org.apache.logging.log4j' + } + implementation 'org.controlsfx:controlsfx:11.1.2' implementation 'org.jsoup:jsoup:1.16.1' @@ -343,8 +348,6 @@ run { 'javafx.controls/com.sun.javafx.scene.control' : 'org.jabref', 'org.controlsfx.controls/impl.org.controlsfx.skin' : 'org.jabref', - 'javafx.controls/com.sun.javafx.scene.control.behavior' : 'com.jfoenix', - // Not sure why we need to restate the controlfx exports // Taken from here: https://github.com/controlsfx/controlsfx/blob/9.0.0/build.gradle#L1 'javafx.graphics/com.sun.javafx.scene' : 'org.controlsfx.controls', @@ -365,12 +368,7 @@ run { 'javafx.controls/com.sun.javafx.scene.control' : 'org.jabref', 'javafx.controls/javafx.scene.control.skin' : 'org.controlsfx.controls', - 'javafx.graphics/javafx.scene' : 'org.controlsfx.controls', - - 'javafx.controls/com.sun.javafx.scene.control.behavior' : 'com.jfoenix', - 'javafx.base/com.sun.javafx.binding' : 'com.jfoenix', - 'javafx.graphics/com.sun.javafx.stage' : 'com.jfoenix', - 'javafx.base/com.sun.javafx.event' : 'com.jfoenix' + 'javafx.graphics/javafx.scene' : 'org.controlsfx.controls' ] } } @@ -588,7 +586,8 @@ jlink { 'org.mariadb.jdbc.credential.env.EnvCredentialPlugin', 'org.mariadb.jdbc.credential.system.PropertiesCredentialPlugin' provides 'java.security.Provider' with 'org.bouncycastle.jce.provider.BouncyCastleProvider', - 'org.bouncycastle.pqc.jcajce.provider.BouncyCastlePQCProvider' } + 'org.bouncycastle.pqc.jcajce.provider.BouncyCastlePQCProvider' + } jpackage { outputDir = "distribution" diff --git a/docs/code-howtos/javafx.md b/docs/code-howtos/javafx.md index c08fe96b489..c5faee31d74 100644 --- a/docs/code-howtos/javafx.md +++ b/docs/code-howtos/javafx.md @@ -201,7 +201,6 @@ JabRef makes heavy use of Properties and Bindings. These are wrappers around Obs * [Validation framework](https://github.com/sialcasa/mvvmFX/wiki/Validation) * [mvvm framework](https://github.com/sialcasa/mvvmFX/wiki) * [CSS Reference](http://docs.oracle.com/javafx/2/api/javafx/scene/doc-files/cssref.html) -* [JFoenix](https://github.com/jfoenixadmin/JFoenix) Material Designs look & feel * [JavaFX Documentation project](https://fxdocs.github.io/docs/html5/index.html): Collected information on JavaFX in a central place * [FXExperience](http://fxexperience.com) JavaFX Links of the week * [Foojay](https://foojay.io) Java and JavaFX tutorials diff --git a/eclipse.gradle b/eclipse.gradle index dab4df54bb6..9d61901d09e 100644 --- a/eclipse.gradle +++ b/eclipse.gradle @@ -25,12 +25,11 @@ eclipse { } def javafxcontrols = entries.find { isJavafxControls(it) }; - javafxcontrols.entryAttributes['add-exports'] = 'javafx.controls/com.sun.javafx.scene.control=org.jabref:javafx.controls/com.sun.javafx.scene.control.behavior=org.jabref:javafx.controls/javafx.scene.control=org.jabref:javafx.controls/com.sun.javafx.scene.control.behavior=com.jfoenix'; + javafxcontrols.entryAttributes['add-exports'] = 'javafx.controls/com.sun.javafx.scene.control=org.jabref:javafx.controls/com.sun.javafx.scene.control.behavior=org.jabref:javafx.controls/javafx.scene.control=org.jabref'; javafxcontrols.entryAttributes['add-opens'] = 'javafx.controls/com.sun.javafx.scene.control=org.jabref:javafx.controls/com.sun.javafx.scene.control.behavior=org.jabref:javafx.controls/javafx.scene.control=org.jabref:javafx.controls/javafx.scene.control.skin=org.controlsfx.controls'; def javafxgraphics = entries.find { isJavafxGraphics(it) }; javafxgraphics.entryAttributes['add-opens'] = 'javafx.graphics/javafx.scene=org.controlsfx.controls'; - javafxgraphics.entryAttributes['add-exports'] = 'javafx.graphics/com.sun.javafx.stage=com.jfoenix'; def javafxbase = entries.find { isJavafxBase(it) }; javafxbase.entryAttributes['add-exports'] = 'javafx.base/com.sun.javafx.event=org.controlsfx.controls:'; diff --git a/external-libraries.md b/external-libraries.md index 0744d3f3c3e..4a75d8eccf3 100644 --- a/external-libraries.md +++ b/external-libraries.md @@ -99,13 +99,6 @@ URL: https://github.com/lemire/javaewah License: Apache-2.0 ``` -```yaml -Id: com.jfoenix:jfoenix -Project: JavaFX MAterial Design Library -URL: https://github.com/jfoenixadmin/JFoenix -License: Apache-2.0 -``` - ```yaml Id: com.konghq.unirest Project: Unirest for Java @@ -539,7 +532,6 @@ 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.214 -com.jfoenix:jfoenix:9.0.10 com.konghq:unirest-java:3.14.1 com.microsoft.azure:applicationinsights-core:2.4.1 com.microsoft.azure:applicationinsights-logging-log4j2:2.4.1 diff --git a/src/main/java/module-info.java b/src/main/java/module-info.java index 91229649901..81ec7d99844 100644 --- a/src/main/java/module-info.java +++ b/src/main/java/module-info.java @@ -9,12 +9,12 @@ // JavaFX requires javafx.base; requires javafx.graphics; - requires javafx.swing; requires javafx.controls; requires javafx.web; requires javafx.fxml; requires afterburner.fx; - requires com.jfoenix; + requires com.dlsc.gemsfx; + uses com.dlsc.gemsfx.TagsField; requires de.saxsys.mvvmfx; requires org.kordamp.ikonli.core; diff --git a/src/main/java/org/jabref/gui/Base.css b/src/main/java/org/jabref/gui/Base.css index 85b0b83ad34..68f96eebf28 100644 --- a/src/main/java/org/jabref/gui/Base.css +++ b/src/main/java/org/jabref/gui/Base.css @@ -1244,10 +1244,10 @@ We want to have a look that matches our icons in the tool-bar */ -fx-background-color: -jr-accent; } -.jfx-color-picker:armed, -.jfx-color-picker:hover, -.jfx-color-picker:focused, -.jfx-color-picker { +.color-picker:armed, +.color-picker:hover, +.color-picker:focused, +.color-picker { -fx-background-color: transparent, transparent, transparent, transparent; -fx-background-radius: 0px; -fx-background-insets: 0px; @@ -1387,6 +1387,27 @@ We want to have a look that matches our icons in the tool-bar */ -fx-pref-height: 30px; -fx-padding: 0px 0px 0px -8px; -fx-margin: 0em; + -fx-border-style: none; +} + +.tags-field { + -fx-pref-height: 30px; + -fx-margin: 0em; + -fx-border-style: none; +} + +.tags-field > .flow-pane > .tag-view { + -fx-background-color: -fx-default-button; + -fx-text-fill: -fx-focused-text-base-color; +} + +.tags-field > .flow-pane > .tag-view:selected { + -fx-background-color: derive(-fx-default-button, -20%); + -fx-text-fill: -fx-focused-text-base-color; +} + +.tags-field-editor { + -fx-border-width: 0; } .searchBar:invalid { diff --git a/src/main/java/org/jabref/gui/documentviewer/PdfDocumentPageViewModel.java b/src/main/java/org/jabref/gui/documentviewer/PdfDocumentPageViewModel.java index 9010ec3379c..607709742b1 100644 --- a/src/main/java/org/jabref/gui/documentviewer/PdfDocumentPageViewModel.java +++ b/src/main/java/org/jabref/gui/documentviewer/PdfDocumentPageViewModel.java @@ -5,8 +5,9 @@ import java.io.IOException; import java.util.Objects; -import javafx.embed.swing.SwingFXUtils; import javafx.scene.image.Image; +import javafx.scene.image.PixelWriter; +import javafx.scene.image.WritableImage; import org.jabref.architecture.AllowedToUseAwt; @@ -51,7 +52,7 @@ public Image render(int width, int height) { try { int resolution = 96; BufferedImage image = renderer.renderImageWithDPI(pageNumber, 2 * resolution, ImageType.RGB); - return SwingFXUtils.toFXImage(resize(image, width, height), null); + return convertToFxImage(resize(image, width, height)); } catch (IOException e) { // TODO: LOG return null; @@ -68,4 +69,19 @@ public double getAspectRatio() { PDRectangle mediaBox = page.getMediaBox(); return mediaBox.getWidth() / mediaBox.getHeight(); } + + // See https://stackoverflow.com/a/57552025/3450689 + private static Image convertToFxImage(BufferedImage image) { + WritableImage wr = null; + if (image != null) { + wr = new WritableImage(image.getWidth(), image.getHeight()); + PixelWriter pw = wr.getPixelWriter(); + for (int x = 0; x < image.getWidth(); x++) { + for (int y = 0; y < image.getHeight(); y++) { + pw.setArgb(x, y, image.getRGB(x, y)); + } + } + } + return wr; + } } diff --git a/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.fxml b/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.fxml index 4b75bbe6311..10d87fdc46e 100644 --- a/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.fxml +++ b/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.fxml @@ -1,8 +1,8 @@ - + - + diff --git a/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.java b/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.java index 3067ee36a11..8c54685b210 100644 --- a/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.java +++ b/src/main/java/org/jabref/gui/fieldeditors/LinkedEntriesEditor.java @@ -1,53 +1,101 @@ package org.jabref.gui.fieldeditors; +import java.util.Comparator; import java.util.stream.Collectors; import javafx.beans.binding.Bindings; import javafx.fxml.FXML; +import javafx.scene.Node; import javafx.scene.Parent; +import javafx.scene.control.ContentDisplay; +import javafx.scene.control.ContextMenu; +import javafx.scene.control.Label; +import javafx.scene.input.MouseButton; import javafx.scene.layout.HBox; +import org.jabref.gui.ClipBoardManager; +import org.jabref.gui.DialogService; +import org.jabref.gui.JabRefDialogService; +import org.jabref.gui.actions.ActionFactory; +import org.jabref.gui.actions.SimpleCommand; +import org.jabref.gui.actions.StandardActions; import org.jabref.gui.autocompleter.SuggestionProvider; +import org.jabref.gui.icon.IconTheme; +import org.jabref.gui.keyboard.KeyBindingRepository; import org.jabref.gui.util.ViewModelListCellFactory; import org.jabref.logic.integrity.FieldCheckers; +import org.jabref.logic.l10n.Localization; import org.jabref.model.database.BibDatabaseContext; import org.jabref.model.entry.BibEntry; import org.jabref.model.entry.ParsedEntryLink; import org.jabref.model.entry.field.Field; import com.airhacks.afterburner.views.ViewLoader; -import com.jfoenix.controls.JFXChip; -import com.jfoenix.controls.JFXChipView; -import com.jfoenix.controls.JFXDefaultChip; +import com.dlsc.gemsfx.TagsField; +import jakarta.inject.Inject; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; public class LinkedEntriesEditor extends HBox implements FieldEditorFX { - @FXML + private static final Logger LOGGER = LoggerFactory.getLogger(LinkedEntriesEditor.class); + + @FXML public TagsField entryLinkField; + + @Inject private DialogService dialogService; + @Inject private ClipBoardManager clipBoardManager; + @Inject private KeyBindingRepository keyBindingRepository; + private final LinkedEntriesEditorViewModel viewModel; - @FXML - private JFXChipView chipView; public LinkedEntriesEditor(Field field, BibDatabaseContext databaseContext, SuggestionProvider suggestionProvider, FieldCheckers fieldCheckers) { - this.viewModel = new LinkedEntriesEditorViewModel(field, suggestionProvider, databaseContext, fieldCheckers); - ViewLoader.view(this) .root(this) .load(); - chipView.setConverter(viewModel.getStringConverter()); - var autoCompletionItemFactory = new ViewModelListCellFactory() - .withText(ParsedEntryLink::getKey); - chipView.getAutoCompletePopup().setSuggestionsCellFactory(autoCompletionItemFactory); - chipView.getAutoCompletePopup().setCellLimit(5); - chipView.getSuggestions().addAll(suggestionProvider.getPossibleSuggestions().stream().map(ParsedEntryLink::new).collect(Collectors.toList())); - - chipView.setChipFactory((view, item) -> { - JFXChip chip = new JFXDefaultChip<>(view, item); - chip.setOnMouseClicked(event -> viewModel.jumpToEntry(item)); - return chip; + this.viewModel = new LinkedEntriesEditorViewModel(field, suggestionProvider, databaseContext, fieldCheckers); + + entryLinkField.setCellFactory(new ViewModelListCellFactory().withText(ParsedEntryLink::getKey)); + // Mind the .collect(Collectors.toList()) as the list needs to be mutable + entryLinkField.setSuggestionProvider(request -> + suggestionProvider.getPossibleSuggestions().stream() + .filter(suggestion -> suggestion.getCitationKey().orElse("").toLowerCase() + .contains(request.getUserText().toLowerCase())) + .map(ParsedEntryLink::new) + .collect(Collectors.toList())); + entryLinkField.setTagViewFactory(this::createTag); + entryLinkField.setConverter(viewModel.getStringConverter()); + entryLinkField.setNewItemProducer(searchText -> viewModel.getStringConverter().fromString(searchText)); + entryLinkField.setMatcher((entryLink, searchText) -> entryLink.getKey().toLowerCase().startsWith(searchText.toLowerCase())); + entryLinkField.setComparator(Comparator.comparing(ParsedEntryLink::getKey)); + entryLinkField.setShowSearchIcon(false); + entryLinkField.getEditor().getStyleClass().clear(); + entryLinkField.getEditor().getStyleClass().add("tags-field-editor"); + + Bindings.bindContentBidirectional(entryLinkField.getTags(), viewModel.linkedEntriesProperty()); + } + + private Node createTag(ParsedEntryLink entryLink) { + Label tagLabel = new Label(); + tagLabel.setText(entryLinkField.getConverter().toString(entryLink)); + tagLabel.setGraphic(IconTheme.JabRefIcons.REMOVE_TAGS.getGraphicNode()); + tagLabel.getGraphic().setOnMouseClicked(event -> entryLinkField.removeTags(entryLink)); + tagLabel.setContentDisplay(ContentDisplay.RIGHT); + tagLabel.setOnMouseClicked(event -> { + if (event.getClickCount() == 2 && event.getButton().equals(MouseButton.PRIMARY)) { + viewModel.jumpToEntry(entryLink); + } }); - Bindings.bindContentBidirectional(chipView.getChips(), viewModel.linkedEntriesProperty()); + ContextMenu contextMenu = new ContextMenu(); + ActionFactory factory = new ActionFactory(keyBindingRepository); + contextMenu.getItems().addAll( + factory.createMenuItem(StandardActions.COPY, new TagContextAction(StandardActions.COPY, entryLink)), + factory.createMenuItem(StandardActions.CUT, new TagContextAction(StandardActions.CUT, entryLink)), + factory.createMenuItem(StandardActions.DELETE, new TagContextAction(StandardActions.DELETE, entryLink)) + ); + tagLabel.setContextMenu(contextMenu); + return tagLabel; } public LinkedEntriesEditorViewModel getViewModel() { @@ -63,4 +111,35 @@ public void bindToEntry(BibEntry entry) { public Parent getNode() { return this; } + + private class TagContextAction extends SimpleCommand { + private final StandardActions command; + private final ParsedEntryLink entryLink; + + public TagContextAction(StandardActions command, ParsedEntryLink entryLink) { + this.command = command; + this.entryLink = entryLink; + } + + @Override + public void execute() { + switch (command) { + case COPY -> { + clipBoardManager.setContent(entryLink.getKey()); + dialogService.notify(Localization.lang("Copied '%0' to clipboard.", + JabRefDialogService.shortenDialogMessage(entryLink.getKey()))); + } + case CUT -> { + clipBoardManager.setContent(entryLink.getKey()); + dialogService.notify(Localization.lang("Copied '%0' to clipboard.", + JabRefDialogService.shortenDialogMessage(entryLink.getKey()))); + entryLinkField.removeTags(entryLink); + } + case DELETE -> + entryLinkField.removeTags(entryLink); + default -> + LOGGER.info("Action {} not defined", command.getText()); + } + } + } } diff --git a/src/main/java/org/jabref/gui/icon/IconTheme.java b/src/main/java/org/jabref/gui/icon/IconTheme.java index 65489ac1e60..c92fea5ac8e 100644 --- a/src/main/java/org/jabref/gui/icon/IconTheme.java +++ b/src/main/java/org/jabref/gui/icon/IconTheme.java @@ -345,11 +345,9 @@ public enum JabRefIcons implements JabRefIcon { KEEP_ON_TOP(MaterialDesignP.PIN), KEEP_ON_TOP_OFF(MaterialDesignP.PIN_OFF_OUTLINE), OPEN_GLOBAL_SEARCH(MaterialDesignO.OPEN_IN_NEW), - + REMOVE_TAGS(MaterialDesignC.CLOSE), ACCEPT_LEFT(MaterialDesignS.SUBDIRECTORY_ARROW_LEFT), - ACCEPT_RIGHT(MaterialDesignS.SUBDIRECTORY_ARROW_RIGHT), - MERGE_GROUPS(MaterialDesignS.SOURCE_MERGE); private final JabRefIcon icon; diff --git a/src/test/java/org/jabref/testutils/interactive/styletester/StyleTester.fxml b/src/test/java/org/jabref/testutils/interactive/styletester/StyleTester.fxml index 2240e83d8bb..69b59a87a7e 100644 --- a/src/test/java/org/jabref/testutils/interactive/styletester/StyleTester.fxml +++ b/src/test/java/org/jabref/testutils/interactive/styletester/StyleTester.fxml @@ -6,6 +6,7 @@ + @@ -35,7 +36,6 @@ - - +