|
@@ -6,8 +6,10 @@ import org.cryptomator.common.LicenseHolder;
|
|
|
import org.cryptomator.common.settings.Settings;
|
|
|
import org.cryptomator.common.settings.UiTheme;
|
|
|
import org.cryptomator.common.vaults.Vault;
|
|
|
-import org.cryptomator.jni.JniException;
|
|
|
-import org.cryptomator.jni.MacApplicationUiAppearance;
|
|
|
+import org.cryptomator.integrations.uiappearance.Theme;
|
|
|
+import org.cryptomator.integrations.uiappearance.UiAppearanceException;
|
|
|
+import org.cryptomator.integrations.uiappearance.UiAppearanceListener;
|
|
|
+import org.cryptomator.integrations.uiappearance.UiAppearanceProvider;
|
|
|
import org.cryptomator.jni.MacApplicationUiState;
|
|
|
import org.cryptomator.jni.MacFunctions;
|
|
|
import org.cryptomator.ui.common.VaultService;
|
|
@@ -42,20 +44,21 @@ public class FxApplication extends Application {
|
|
|
private final Provider<UnlockComponent.Builder> unlockWindowBuilderProvider;
|
|
|
private final Provider<QuitComponent.Builder> quitWindowBuilderProvider;
|
|
|
private final Optional<MacFunctions> macFunctions;
|
|
|
+ private final Optional<UiAppearanceProvider> appearanceProvider;
|
|
|
private final VaultService vaultService;
|
|
|
private final LicenseHolder licenseHolder;
|
|
|
private final BooleanBinding hasVisibleStages;
|
|
|
-
|
|
|
- private Optional<String> macApperanceObserverIdentifier = Optional.empty();
|
|
|
+ private final UiAppearanceListener systemInterfaceThemeListener = this::systemInterfaceThemeChanged;
|
|
|
|
|
|
@Inject
|
|
|
- FxApplication(Settings settings, Lazy<MainWindowComponent> mainWindow, Lazy<PreferencesComponent> preferencesWindow, Provider<UnlockComponent.Builder> unlockWindowBuilderProvider, Provider<QuitComponent.Builder> quitWindowBuilderProvider, Optional<MacFunctions> macFunctions, VaultService vaultService, LicenseHolder licenseHolder, ObservableSet<Stage> visibleStages) {
|
|
|
+ FxApplication(Settings settings, Lazy<MainWindowComponent> mainWindow, Lazy<PreferencesComponent> preferencesWindow, Provider<UnlockComponent.Builder> unlockWindowBuilderProvider, Provider<QuitComponent.Builder> quitWindowBuilderProvider, Optional<MacFunctions> macFunctions, Optional<UiAppearanceProvider> appearanceProvider, VaultService vaultService, LicenseHolder licenseHolder, ObservableSet<Stage> visibleStages) {
|
|
|
this.settings = settings;
|
|
|
this.mainWindow = mainWindow;
|
|
|
this.preferencesWindow = preferencesWindow;
|
|
|
this.unlockWindowBuilderProvider = unlockWindowBuilderProvider;
|
|
|
this.quitWindowBuilderProvider = quitWindowBuilderProvider;
|
|
|
this.macFunctions = macFunctions;
|
|
|
+ this.appearanceProvider = appearanceProvider;
|
|
|
this.vaultService = vaultService;
|
|
|
this.licenseHolder = licenseHolder;
|
|
|
this.hasVisibleStages = Bindings.isNotEmpty(visibleStages);
|
|
@@ -67,7 +70,7 @@ public class FxApplication extends Application {
|
|
|
|
|
|
EasyBind.subscribe(hasVisibleStages, this::hasVisibleStagesChanged);
|
|
|
|
|
|
- settings.theme().addListener(this::themeChanged);
|
|
|
+ settings.theme().addListener(this::appThemeChanged);
|
|
|
loadSelectedStyleSheet(settings.theme().get());
|
|
|
}
|
|
|
|
|
@@ -116,45 +119,60 @@ public class FxApplication extends Application {
|
|
|
return vaultService;
|
|
|
}
|
|
|
|
|
|
- private void themeChanged(@SuppressWarnings("unused") ObservableValue<? extends UiTheme> observable, @SuppressWarnings("unused") UiTheme oldValue, UiTheme newValue) {
|
|
|
- if (macApperanceObserverIdentifier.isPresent()) {
|
|
|
- macFunctions.map(MacFunctions::uiAppearance).ifPresent(uiAppearance -> uiAppearance.removeListener(macApperanceObserverIdentifier.get()));
|
|
|
- macApperanceObserverIdentifier = Optional.empty();
|
|
|
- }
|
|
|
+ private void appThemeChanged(@SuppressWarnings("unused") ObservableValue<? extends UiTheme> observable, @SuppressWarnings("unused") UiTheme oldValue, UiTheme newValue) {
|
|
|
+ appearanceProvider.ifPresent(appearanceProvider -> {
|
|
|
+ try {
|
|
|
+ appearanceProvider.removeListener(systemInterfaceThemeListener);
|
|
|
+ } catch (UiAppearanceException e) {
|
|
|
+ LOG.error("Failed to disable automatic theme switching.");
|
|
|
+ }
|
|
|
+ });
|
|
|
loadSelectedStyleSheet(newValue);
|
|
|
}
|
|
|
|
|
|
private void loadSelectedStyleSheet(UiTheme desiredTheme) {
|
|
|
UiTheme theme = licenseHolder.isValidLicense() ? desiredTheme : UiTheme.LIGHT;
|
|
|
switch (theme) {
|
|
|
- case LIGHT -> setToLightTheme();
|
|
|
- case DARK -> setToDarkTheme();
|
|
|
+ case LIGHT -> applyLightTheme();
|
|
|
+ case DARK -> applyDarkTheme();
|
|
|
case AUTOMATIC -> {
|
|
|
- macFunctions.map(MacFunctions::uiAppearance).ifPresent(uiAppearance -> {
|
|
|
- macApperanceObserverIdentifier = Optional.of(uiAppearance.addListener(this::macInterfaceThemeChanged));
|
|
|
+ appearanceProvider.ifPresent(appearanceProvider -> {
|
|
|
+ try {
|
|
|
+ appearanceProvider.addListener(systemInterfaceThemeListener);
|
|
|
+ } catch (UiAppearanceException e) {
|
|
|
+ LOG.error("Failed to enable automatic theme switching.");
|
|
|
+ }
|
|
|
});
|
|
|
- macInterfaceThemeChanged();
|
|
|
+ applySystemTheme();
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- private void macInterfaceThemeChanged() {
|
|
|
- macFunctions.map(MacFunctions::uiAppearance).ifPresent(uiAppearance -> {
|
|
|
- switch (uiAppearance.getCurrentInterfaceStyle()) {
|
|
|
- case LIGHT -> setToLightTheme();
|
|
|
- case DARK -> setToDarkTheme();
|
|
|
- }
|
|
|
+ private void systemInterfaceThemeChanged(Theme theme) {
|
|
|
+ switch (theme) {
|
|
|
+ case LIGHT -> applyLightTheme();
|
|
|
+ case DARK -> applyDarkTheme();
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ private void applySystemTheme() {
|
|
|
+ appearanceProvider.ifPresent(appearanceProvider -> {
|
|
|
+ systemInterfaceThemeChanged(appearanceProvider.getSystemTheme());
|
|
|
});
|
|
|
}
|
|
|
|
|
|
- private void setToLightTheme() {
|
|
|
+ private void applyLightTheme() {
|
|
|
Application.setUserAgentStylesheet(getClass().getResource("/css/light_theme.css").toString());
|
|
|
- macFunctions.map(MacFunctions::uiAppearance).ifPresent(JniException.ignore(MacApplicationUiAppearance::setToAqua));
|
|
|
+ appearanceProvider.ifPresent(appearanceProvider -> {
|
|
|
+ appearanceProvider.adjustToTheme(Theme.LIGHT);
|
|
|
+ });
|
|
|
}
|
|
|
|
|
|
- private void setToDarkTheme() {
|
|
|
+ private void applyDarkTheme() {
|
|
|
Application.setUserAgentStylesheet(getClass().getResource("/css/dark_theme.css").toString());
|
|
|
- macFunctions.map(MacFunctions::uiAppearance).ifPresent(JniException.ignore(MacApplicationUiAppearance::setToDarkAqua));
|
|
|
+ appearanceProvider.ifPresent(appearanceProvider -> {
|
|
|
+ appearanceProvider.adjustToTheme(Theme.DARK);
|
|
|
+ });
|
|
|
}
|
|
|
|
|
|
}
|