From d83de4265ddfc05e3dacfd0a915aee50a010db71 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bartosz=20Korczy=C5=84ski?= Date: Sat, 9 Mar 2024 03:29:33 +0000 Subject: [PATCH] Avalonia 11 (#214) The editor has been updated to the latest UI framework version. The new version has many breaking changes, if you think there is some new bug, please report it. --- AntlrSupport.props | 6 +- Avalonia.props | 16 +- AvaloniaEdit | 2 +- .../AvaloniaStyles.Desktop.csproj | 22 + AvaloniaStyles.Desktop/Program.cs | 16 + AvaloniaStyles/App.xaml | 4 + AvaloniaStyles/App.xaml.cs | 20 +- AvaloniaStyles/AvaloniaStyles.csproj | 15 +- .../Controls/AccentSolidColorBrush.cs | 123 +- .../Controls/AlternativeScrollViewer.cs | 37 - .../Controls/BaseMessageBoxWindow.cs | 19 +- .../Controls/CompletionComboBox.axaml | 6 +- .../Controls/CompletionComboBox.axaml.cs | 105 +- AvaloniaStyles/Controls/ControlRenderer.cs | 89 ++ .../Controls/DismissPopupBehaviour.cs | 2 +- AvaloniaStyles/Controls/DropDownButton.axaml | 23 - .../Controls/DropDownButton.axaml.cs | 37 - AvaloniaStyles/Controls/ExtendedWindow.cs | 55 +- AvaloniaStyles/Controls/Extensions.cs | 58 +- .../Controls/FastTableView/ColorsDark.axaml | 4 +- .../Controls/FastTableView/ColorsLight.axaml | 4 +- .../Controls/FastTableView/Extensions.cs | 5 +- .../FastTableView/ICustomCellDrawer.cs | 33 +- .../Controls/FastTableView/PhantomTextBox.cs | 17 +- AvaloniaStyles/Controls/FastTableView/Row.cs | 3 - .../VeryFastTableView.Arrange.cs | 2 +- .../FastTableView/VeryFastTableView.Data.cs | 11 +- .../VeryFastTableView.Navigation.cs | 28 +- .../VeryFastTableView.Properties.cs | 16 +- .../VeryFastTableView.Rendering.cs | 48 +- .../FastTableView/VeryFastTableView.cs | 128 +- .../Controls/FastTableView/WdeCellDrawer.cs | 14 +- AvaloniaStyles/Controls/GridView.cs | 132 +- AvaloniaStyles/Controls/GridViewStyle.axaml | 119 +- AvaloniaStyles/Controls/GroupingListBox.axaml | 15 +- .../Controls/GroupingListBox.axaml.cs | 75 +- .../Controls/HamburgerMenuButton.axaml | 55 + .../Controls/HamburgerMenuButton.axaml.cs | 9 + .../Controls/IconElement/BitmapIcon.cs | 88 +- .../Controls/IconElement/BitmapIconSource.cs | 7 +- .../Controls/IconElement/FAIconElement.cs | 5 +- .../Controls/IconElement/FAPathIcon.cs | 4 +- .../IconElement/FAPathIcon.properties.cs | 2 +- .../Controls/IconElement/FontIcon.cs | 22 +- .../Controls/IconElement/IconSourceElement.cs | 2 +- .../Controls/IconElement/ImageIcon.cs | 5 +- .../Controls/IconElement/PathIconSource.cs | 6 +- .../Controls/IconElement/SymbolIcon.cs | 11 +- AvaloniaStyles/Controls/InfoBar/InfoBar.cs | 8 +- .../Controls/LightSolidColorBrush.cs | 42 - AvaloniaStyles/Controls/ManagedMenu.cs | 23 - .../NumberIndicator.axaml | 2 +- .../NumberIndicator.axaml.cs | 2 - .../BaseVirtualizedTableController.cs | 35 +- ...VirtualizedVeryFastTableView.Navigation.cs | 29 +- .../VirtualizedVeryFastTableView.Rendering.cs | 92 +- .../VirtualizedVeryFastTableView.cs | 88 +- AvaloniaStyles/Controls/RecyclableViewList.cs | 13 +- AvaloniaStyles/Controls/SettingItem.axaml | 10 +- AvaloniaStyles/Controls/SettingItem.axaml.cs | 2 +- AvaloniaStyles/Controls/StretchStackPanel.cs | 26 +- AvaloniaStyles/Controls/StretchWrapPanel.cs | 8 +- AvaloniaStyles/Controls/ToolbarControl.axaml | 137 +- .../Controls/ToolbarControl.axaml.cs | 78 - .../Controls/ToolbarItemsControl.cs | 67 + AvaloniaStyles/Controls/ToolbarPanel.cs | 39 +- AvaloniaStyles/Controls/ToolsTabControl.cs | 127 -- AvaloniaStyles/Controls/TwoColumnsPanel.cs | 6 +- .../VirtualizedGridCheckBox.cs | 4 +- .../VirtualizedGridView.cs | 52 +- .../VirtualizedGridViewItemPresenter.cs | 9 +- .../VirtualizedGridViewStyle.axaml | 88 +- AvaloniaStyles/Controls/WizardPanel.axaml | 4 +- AvaloniaStyles/Controls/WizardPanel.axaml.cs | 10 +- .../Converters/ChromeToBoolConverter.cs | 2 +- .../Converters/IListCountToBoolConverter.cs | 6 +- .../Converters/InverseBoolConverter.cs | 6 +- .../IsCheckedToCheckBoxConverter.cs | 2 +- .../Converters/ToStringConverter.cs | 2 +- AvaloniaStyles/Demo/ButtonImageDemo.cs | 8 +- AvaloniaStyles/Demo/DemoDataContext.cs | 52 + AvaloniaStyles/Demo/MainView.axaml | 79 + AvaloniaStyles/Demo/MainView.axaml.cs | 28 + AvaloniaStyles/Demo/MainWindow.axaml | 159 +- AvaloniaStyles/Demo/MainWindow.axaml.cs | 51 +- AvaloniaStyles/Demo/MessageBox.axaml | 14 +- AvaloniaStyles/Demo/MessageBox.axaml.cs | 8 +- AvaloniaStyles/Styles/BigSur/Button.xaml | 19 - AvaloniaStyles/Styles/BigSur/CheckBox.xaml | 2 +- .../Styles/BigSur/CompletionComboBox.axaml | 2 +- AvaloniaStyles/Styles/BigSur/ContextMenu.xaml | 4 +- .../Styles/BigSur/GroupingListBox.axaml | 2 +- AvaloniaStyles/Styles/BigSur/StatusBar.xaml | 2 +- AvaloniaStyles/Styles/Catalina/Button.xaml | 13 - AvaloniaStyles/Styles/Catalina/CheckBox.xaml | 2 +- .../Styles/Catalina/CompletionComboBox.axaml | 2 +- .../Styles/Catalina/ContextMenu.xaml | 4 +- .../Styles/Catalina/GroupingListBox.axaml | 2 +- .../Styles/{ => MacOs}/MacOsBigSurDark.xaml | 0 .../Styles/{ => MacOs}/MacOsBigSurLight.xaml | 0 .../Styles/{ => MacOs}/MacOsCatalinaDark.xaml | 0 .../{ => MacOs}/MacOsCatalinaLight.xaml | 0 .../Styles/StyleIncludeColorAware.cs | 5 +- AvaloniaStyles/Styles/Windows10/Button.axaml | 11 +- .../Styles/Windows10/CheckBox.axaml | 63 +- .../Styles/Windows10/ColorsDark.axaml | 156 ++ .../Styles/Windows10/ColorsDark.xaml | 106 -- .../{ColorsLight.xaml => ColorsLight.axaml} | 49 +- .../Styles/Windows10/CompletionComboBox.axaml | 8 +- AvaloniaStyles/Styles/Windows10/Dock.axaml | 23 + .../Styles/Windows10/DockLight.axaml | 14 +- .../Windows10/{ListBox.xaml => ListBox.axaml} | 0 .../Styles/Windows10/NativeMenuBar.xaml | 26 - .../Windows10/{SideBar.xaml => SideBar.axaml} | 0 .../{StatusBar.xaml => StatusBar.axaml} | 8 +- .../Windows10/{Style.xaml => Style.axaml} | 27 +- .../Styles/Windows10/TabControl.axaml | 27 +- .../{TabStrip.xaml => TabStrip.axaml} | 4 +- .../Styles/Windows10/ToolbarPanel.axaml | 98 +- .../Styles/Windows10/ToolsTabControl.xaml | 88 -- .../Styles/Windows10/TreeViewItem.axaml | 46 +- AvaloniaStyles/Styles/Windows10/Window.axaml | 159 ++ AvaloniaStyles/Styles/Windows10/Window.xaml | 164 -- .../Styles/Windows10/WindowMessageBox.axaml | 117 ++ ...Windows10Dark.xaml => Windows10Dark.axaml} | 6 +- ...ndows10Light.xaml => Windows10Light.axaml} | 7 +- AvaloniaStyles/Styles/Windows10Light.cs | 12 + AvaloniaStyles/Styles/Windows11/Button.axaml | 140 +- .../{ColorsDark.xaml => ColorsDark.axaml} | 34 +- .../{ColorsLight.xaml => ColorsLight.axaml} | 40 +- .../Styles/Windows11/DockLight.axaml | 22 +- .../Styles/Windows11/NativeMenuBar.xaml | 48 - .../Windows11/{Style.xaml => Style.axaml} | 33 +- .../Styles/Windows11/TabControl.axaml | 2 +- AvaloniaStyles/Styles/Windows11/TextBox.axaml | 269 ++-- AvaloniaStyles/Styles/Windows11/Window.axaml | 28 +- ...Windows11Dark.xaml => Windows11Dark.axaml} | 6 +- ...ndows11Light.xaml => Windows11Light.axaml} | 6 +- AvaloniaStyles/SystemTheme.cs | 16 +- AvaloniaStyles/Utils/HslColor.cs | 5 +- AvaloniaStyles/Utils/SolidBrushTransition.cs | 156 -- AvaloniaStyles/Utils/Win32.cs | 5 +- BaseDesktopLoader/BaseDesktopLoader.csproj | 31 + BaseDesktopLoader/Program.cs | 113 ++ DBCD | 2 +- DatabaseTester/ConsoleMessageBoxService.cs | 3 - DatabaseTester/DatabaseTester.csproj | 4 +- DatabaseTester/Program.cs | 6 +- Directory.Build.props | 18 +- Dock | 2 +- GeminiGraphEditor/GeminiGraphEditor.csproj | 2 +- LoaderAvalonia.Web/AppBundle/Logo.svg | 5 + LoaderAvalonia.Web/AppBundle/app.css | 74 + LoaderAvalonia.Web/AppBundle/favicon.ico | Bin 0 -> 176111 bytes LoaderAvalonia.Web/AppBundle/index.html | 30 + LoaderAvalonia.Web/AppBundle/main.js | 31 + LoaderAvalonia.Web/LoaderAvalonia.Web.csproj | 116 ++ LoaderAvalonia.Web/NullDebuggerService.cs | 82 + LoaderAvalonia.Web/NullGameView.cs | 16 + .../NullRemoteConnectorService.cs | 36 + .../NullSourceCodePathService.cs | 13 + LoaderAvalonia.Web/NullUpdateViewModel.cs | 16 + .../NullVisualStudioManagerViewModel.cs | 11 + LoaderAvalonia.Web/Program.cs | 77 + .../Properties/launchSettings.json | 13 + LoaderAvalonia.Web/WebModule.cs | 8 + .../runtimeconfig.template.json | 11 + LoaderAvalonia.iOS/AppDelegate.cs | 20 + LoaderAvalonia.iOS/Entitlements.plist | 5 + LoaderAvalonia.iOS/Info.plist | 47 + LoaderAvalonia.iOS/LoaderAvalonia.iOS.csproj | 51 + LoaderAvalonia.iOS/Main.cs | 69 + LoaderAvalonia.iOS/NullDebuggerService.cs | 82 + LoaderAvalonia.iOS/NullGameView.cs | 16 + .../NullRemoteConnectorService.cs | 36 + .../NullSourceCodePathService.cs | 13 + LoaderAvalonia.iOS/NullUpdateViewModel.cs | 16 + .../NullVisualStudioManagerViewModel.cs | 11 + LoaderAvalonia.iOS/Resources/LaunchScreen.xib | 43 + LoaderAvalonia.iOS/iOSModule.cs | 8 + LoaderAvalonia/LoaderAvalonia.csproj | 12 +- LoaderAvalonia/Program.cs | 21 +- Modules/AvaloniaGraph/AvaloniaGraph.csproj | 4 +- Modules/AvaloniaGraph/Controls/BezierLine.cs | 24 +- .../AvaloniaGraph/Controls/ConnectionItem.cs | 2 +- .../Controls/ConnectionsContainer.cs | 16 +- .../AvaloniaGraph/Controls/ConnectorItem.cs | 5 +- .../ConnectionDragCompletedEventArgs.cs | 2 +- .../Events/ConnectionDragEventArgs.cs | 19 +- .../Events/ConnectionDraggingEventArgs.cs | 2 +- .../Events/ConnectorItemBaseEventArgs.cs | 16 +- .../ConnectorItemDragCompletedEventArgs.cs | 2 +- .../Events/ConnectorItemDraggingEventArgs.cs | 2 +- .../AvaloniaGraph/Controls/GraphControl.cs | 20 +- .../Controls/GraphNodeItemView.cs | 28 +- .../AvaloniaGraph/Controls/NodesContainer.cs | 16 +- Modules/AvaloniaGraph/Style.axaml | 26 +- Modules/CrashReport/App.axaml | 5 +- Modules/CrashReport/AvaloniaViewLocator.cs | 5 +- Modules/CrashReport/CrashReport.csproj | 5 +- Modules/CrashReport/CrashReportView.axaml | 7 +- Modules/CrashReport/CrashReportViewModel.cs | 71 +- Modules/CrashReport/MainWindowViewModel.cs | 22 +- Modules/CrashReport/NoCrashView.axaml | 4 +- Modules/CrashReport/Program.cs | 4 +- .../Services/CommentPublisherService.cs | 3 +- .../Views/TimelineView.axaml | 4 +- .../Views/TimelineView.axaml.cs | 2 +- .../WDE.AnniversaryInfo.csproj | 4 +- .../DatabaseDefinitionEditorModule.cs | 7 +- .../ViewModels/DefinitionEditorViewModel.cs | 5 +- ...SourceTreeTableDefinitionEditorProvider.cs | 3 +- .../Views/ColumnView.axaml | 4 +- .../Views/ColumnsView.axaml | 4 +- .../Views/CommandView.axaml | 2 +- .../Views/CommandsView.axaml | 2 +- .../Views/ConditionsView.axaml | 2 +- .../Controls/DatabaseColumnCompletionBox.cs | 6 +- .../DatabaseTableCommandCompletionBox.cs | 4 +- .../Controls/DatabaseTableCompletionBox.cs | 6 +- .../Views/Controls/KeyBindingBox.cs | 6 +- .../NullableDatabaseColumnCompletionBox.cs | 2 +- .../Views/Controls/OnReleaseListBoxItem.cs | 19 +- .../Views/DefinitionEditorView.axaml | 12 +- .../Views/DefinitionToolView.axaml | 8 +- .../Views/DefinitionView.axaml | 12 +- .../Views/ForeignTables.axaml | 4 +- .../Views/TemplateModePreview.axaml | 4 +- .../WDE.DatabaseDefinitionEditor.csproj | 4 +- .../Services/QueryParserServiceTests.cs | 4 +- .../WDE.DatabaseEditors.Test.csproj | 8 +- .../Services/DebuggerServiceTests.cs | 3 +- .../WDE.Debugger.Test.csproj | 10 +- .../WDE.Debugger/Services/DebuggerService.cs | 17 +- .../DebugPointsInspectorToolBar.axaml | 2 +- .../Inspector/EditDebugPointPopup.axaml.cs | 2 +- Modules/WDE.Debugger/WDE.Debugger.csproj | 8 +- .../EventScriptDataJsonProvider.cs | 11 +- .../EventScriptDataProvider.cs | 24 +- .../IEventScriptDataJsonProvider.cs | 4 +- .../Views/EventScriptViewerView.axaml | 2 +- .../Views/EventScriptViewerView.axaml.cs | 2 +- .../WDE.EventScriptsEditor.csproj | 6 +- .../Views/FirstTimeWizardView.axaml | 4 +- .../Views/FirstTimeWizardView.axaml.cs | 2 +- .../WDE.FirstTimeWizard.csproj | 4 +- .../WDE.HttpDatabase/HttpDatabaseModule.cs | 29 + .../WDE.HttpDatabase/HttpDatabaseProvider.cs | 40 + .../HttpDatabaseProviderImpl.cs | 1390 +++++++++++++++++ .../MockHotfixDatabaseSettingsProvider.cs | 35 + .../Models/AzerothMySqlSmartScriptLine.cs | 220 +++ .../Models/AzerothMySqlSpellDbc.cs | 15 + .../Models/JsonAreaTriggerCreateProperties.cs | 15 + .../Models/JsonAzerothString.cs | 44 + .../Models/JsonConversationActor.cs | 11 + .../Models/JsonConversationActorTemplate.cs | 9 + .../Models/JsonCoreCommandHelp.cs | 15 + .../JsonCreatureTemplateAndDifficulty.cs | 12 + .../Models/JsonLootTemplateName.cs | 10 + .../Models/JsonNpcSpellClickSpell.cs | 29 + .../Models/JsonSpawnGroupFormation.cs | 14 + .../Models/JsonTrinityString.cs | 44 + .../Models/MySqlAreaTriggerScript.cs | 15 + .../Models/MySqlAreaTriggerTemplate.cs | 19 + .../Models/MySqlBroadcastText.cs | 99 ++ .../Models/MySqlBroadcastTextLocale.cs | 21 + .../Models/MySqlConditionLine.cs | 68 + .../Models/MySqlConversationTemplate.cs | 18 + .../Models/MySqlCreatureAddon.cs | 191 +++ .../Models/MySqlCreatureClassLevelStat.cs | 67 + .../Models/MySqlCreatureEquipmentTemplate.cs | 40 + .../Models/MySqlCreatureModelInfo.cs | 33 + .../Models/MySqlCreatureTemplateDifficulty.cs | 25 + .../Models/MySqlCreatureTemplateWrath.cs | 308 ++++ .../Models/MySqlCreatureText.cs | 48 + .../Models/MySqlCreatureWrath.cs | 198 +++ .../Models/MySqlEventScriptBaseLine.cs | 123 ++ .../WDE.HttpDatabase/Models/MySqlGameEvent.cs | 15 + .../Models/MySqlGameEventCreature.cs | 21 + .../Models/MySqlGameEventGameObject.cs | 21 + .../Models/MySqlGameObjectTemplate.cs | 52 + .../Models/MySqlGameObjectWrath.cs | 123 ++ .../Models/MySqlGossipMenu.cs | 23 + .../Models/MySqlGossipMenuLine.cs | 23 + .../Models/MySqlGossipMenuOptionWrath.cs | 103 ++ .../Models/MySqlItemTemplate.cs | 18 + .../WDE.HttpDatabase/Models/MySqlLootEntry.cs | 24 + .../WDE.HttpDatabase/Models/MySqlNpcText.cs | 20 + .../WDE.HttpDatabase/Models/MySqlPhaseName.cs | 17 + .../Models/MySqlPlayerChoice.cs | 32 + .../Models/MySqlPointOfInterest.cs | 33 + .../Models/MySqlQuestObjective.cs | 33 + .../Models/MySqlQuestRequestItem.cs | 24 + .../Models/MySqlQuestTemplate.cs | 142 ++ .../Models/MySqlQuestTemplateAddon.cs | 55 + .../Models/MySqlSceneTemplate.cs | 21 + .../Models/MySqlSmartScriptLine.cs | 241 +++ .../Models/MySqlSpawnGroupSpawn.cs | 24 + .../Models/MySqlSpawnGroupTemplate.cs | 24 + .../Models/MySqlSpellScriptName.cs | 15 + .../Models/RbacLinkedPermission.cs | 15 + .../WDE.HttpDatabase/Models/RbacPermission.cs | 15 + .../Models/TrinityMySqlSpellDbc.cs | 25 + Modules/WDE.HttpDatabase/Models/Waypoints.cs | 194 +++ .../QueryExecutor/DummyAuthMySqlExecutor.cs | 38 + .../QueryExecutor/DummyHotfixMySqlExecutor.cs | 33 + .../QueryExecutor/HttpMySqlExecutor.cs | 140 ++ .../QueryExecutor/MySqlType.cs | 19 + .../QueryExecutor/SelectResult.cs | 8 + .../WDE.HttpDatabase/WDE.HttpDatabase.csproj | 24 + .../Configuration/Views/CustomCellDrawer.cs | 3 +- .../Views/LootEditorConfigurationView.axaml | 12 +- .../LootCrossReferencesView.axaml | 2 +- .../LootCrossReferencesViewModel.cs | 2 +- .../WDE.LootEditor/DataLoaders/LootLoader.cs | 5 +- .../StandaloneLootEditorToolBar.axaml | 34 +- .../Editor/ViewModels/LootEditorViewModel.cs | 11 +- .../Editor/ViewModels/LootItemViewModel.cs | 2 +- .../Editor/Views/CustomCellDrawer.cs | 5 +- .../Editor/Views/LootEditorToolBar.axaml.cs | 16 +- .../Editor/Views/LootEditorView.axaml | 4 +- .../Editor/Views/LootEditorView.axaml.cs | 3 +- .../Picker/Views/CustomCellDrawer.cs | 3 +- .../Picker/Views/LootPickerView.axaml | 4 +- .../Picker/Views/LootPickerView.axaml.cs | 4 +- Modules/WDE.LootEditor/WDE.LootEditor.csproj | 4 +- .../Models/GameEventService.cs | 45 +- .../Rendering/Modules/PathEditor.cs | 2 +- .../Rendering/QuickSpawnMenu.cs | 32 +- .../Rendering/SpawnGizmo.cs | 1 - .../Tools/Views/SpawnGroupPickerView.axaml | 2 +- .../Tools/Views/SpawnGroupPickerView.axaml.cs | 2 +- .../ViewModels/CreatureSpawnInstance.cs | 2 +- .../ViewModels/GameObjectSpawnInstance.cs | 2 +- .../Views/SpawnsFastTreeList.cs | 47 +- .../Views/SpawnsQueriesToolView.axaml.cs | 2 +- .../Views/SpawnsToolView.axaml | 4 +- .../Views/SpawnsToolView.axaml.cs | 14 +- .../WDE.MapSpawnsEditor.csproj | 4 +- .../Utils/IntervalTrees/IntervalTreeTests.cs | 66 +- .../OptimizedIntervalTreeTests.cs | 14 +- .../WDE.PacketViewer.Test.csproj | 15 +- .../Views/PathPreviewToolBar.axaml | 2 +- .../Views/PathPreviewToolBar.axaml.cs | 2 +- .../Views/PathPreviewView.axaml.cs | 2 +- .../WDE.PathPreviewTool.csproj | 4 +- .../Services/InterEditorIntegration.cs | 5 +- .../WDE.Profiles/Services/ProfileService.cs | 10 +- .../SocketBasedEditorCommunication.cs | 14 +- Modules/WDE.Profiles/WDE.Profiles.csproj | 4 +- .../Base/QueryGenerator.cs | 3 +- .../TrinityCreatureDiffQueryProvider.cs | 6 +- .../TrinityGameObjectDiffQueryProvider.cs | 6 +- .../WDE.QueryGenerators.csproj | 4 +- .../WDE.QuestChainEditor.Test.csproj | 8 +- .../WDE.QuestChainEditor.csproj | 2 +- .../Sessions/SessionSerializerDeserializer.cs | 4 +- .../WDE.Sessions/Sessions/SessionService.cs | 33 +- Modules/WDE.Sessions/WDE.Sessions.csproj | 8 +- .../ConnectionListToolViewModelTests.cs | 12 +- .../SqlWorkbenchViewModelTests.cs | 40 +- .../Mock/MockRawMySqlConnection.cs | 6 +- .../WDE.SqlWorkbench.Test.csproj | 12 +- .../Models/Data/PublicMySqlDecimal.cs | 12 + .../MariaDb/MariaDownloadView.axaml | 2 +- .../Downloaders/MySql/MySqlDownloadView.axaml | 2 +- .../LanguageServer/SqlsLanguageServer.cs | 5 +- .../LanguageServer/SqlsLanguageServerFile.cs | 19 +- .../QuerySplitter/MySQLParserServicesImpl.cs | 2 +- .../QuerySplitter/ThreadedQuerySplitter.cs | 3 +- .../Services/QueryUtils/QueryUtility.cs | 3 +- .../Services/SqlEditorService.cs | 1 - .../SyntaxValidator/AntlrSyntaxValidator.cs | 6 +- .../TablesPanel/ConnectionListToolView.axaml | 4 +- .../ConnectionListToolView.axaml.cs | 6 +- .../Services/TextMarkers/TextMarkerService.cs | 8 +- .../UserQuestions/IUserQuestionsService.cs | 5 +- .../Settings/ConnectionConfigViewModel.cs | 1 + .../SqlWorkbenchConfigurationView.axaml | 8 +- .../ViewModels/ActionsOutputViewModel.cs | 7 +- .../ViewModels/DumpViewModel.cs | 5 +- .../ViewModels/EditorCompletionData.cs | 3 +- .../ViewModels/SelectResultsViewModel.cs | 8 +- .../ViewModels/SelectSingleTableViewModel.cs | 7 +- .../ViewModels/SqlWorkbenchViewModel.cs | 13 +- .../Views/ConnectionsStatusBarItemView.axaml | 4 +- Modules/WDE.SqlWorkbench/Views/DumpView.axaml | 5 +- .../Views/SelectSingleTableView.axaml.cs | 3 +- .../Views/SqlWorkbenchToolBar.axaml | 2 +- .../Views/SqlWorkbenchView.axaml | 6 +- .../Views/SqlWorkbenchView.axaml.cs | 4 +- .../Views/TableCreatorView.axaml | 8 +- .../WDE.SqlWorkbench/WDE.SqlWorkbench.csproj | 9 +- Prism.Avalonia | 2 +- QuestChainTest/QuestChainTest.csproj | 2 +- README.md | 6 +- Rendering/AvaloniaRenderingTester/App.axaml | 2 +- .../AvaloniaRenderingTester.csproj | 6 +- .../MainWindow.axaml.cs | 8 +- .../OpenGLBindings/OpenGLBindings.csproj | 2 +- .../RenderingTester/GameStandaloneWindow.cs | 4 +- Rendering/RenderingTester/OpenTKDevice.cs | 551 +++++++ .../RenderingTester/RenderingTester.csproj | 6 +- .../SingleThreadSynchronizationContext.cs | 2 +- .../TheEngineOpenTkWindow.cs | 2 +- Rendering/RenderingTester/Window.cs | 2 +- Rendering/TheAvaloniaOpenGL/Device.cs | 752 +++++---- .../NativeOpenGlControlBase.cs | 49 +- Rendering/TheAvaloniaOpenGL/OpenGlBase.cs | 530 +++---- Rendering/TheAvaloniaOpenGL/OpenGlBase2.cs | 328 +--- .../TheAvaloniaOpenGL/OpenTKGlControl2.cs | 58 +- .../TheAvaloniaOpenGL/RealDeviceWrapper.cs | 6 +- .../Resources/NativeBuffer.cs | 2 +- .../TheAvaloniaOpenGL/Resources/Sampler.cs | 2 +- .../Resources/TextureArray.cs | 1 - .../TheAvaloniaOpenGL.csproj | 15 +- Rendering/TheAvaloniaOpenGL/TheDevice.cs | 1 + .../Coroutines/CoroutineManager.cs | 2 +- .../ECS/EntityDataMangerTests.cs | 14 +- .../TheEngine.Test/ECS/EntityManagerTests.cs | 2 +- .../ECS/EntityPerformanceTests.cs | 4 +- .../TheEngine.Test/TheEngine.Test.csproj | 10 +- Rendering/TheEngine/Data/ObjModel.cs | 2 +- Rendering/TheEngine/Engine.cs | 5 - Rendering/TheEngine/Entities/Mesh.cs | 2 - Rendering/TheEngine/Managers/RenderManager.cs | 4 +- Rendering/TheEngine/Managers/ShaderManager.cs | 1 - .../TheEngine/Managers/TextureManager.cs | 14 +- Rendering/TheEngine/Managers/UIManager.cs | 1 - Rendering/TheEngine/NativeTheEnginePanel.cs | 8 +- Rendering/TheEngine/TheEngine.csproj | 10 +- Rendering/TheEngine/TheEnginePanel.cs | 93 +- Rendering/TheEngine/Utils/NaiveThreadPool.cs | 6 +- Rendering/TheMaths/TheMaths.csproj | 4 +- Rendering/TheMaths/Utilities.cs | 34 +- .../WDE.MapSpawns/Models/GameEventService.cs | 14 +- .../WDE.MapSpawns/Rendering/SpawnGizmo.cs | 1 - .../WDE.MapSpawns/Views/SpawnsFastTreeList.cs | 19 +- .../WDE.MapSpawns/Views/SpawnsToolView.axaml | 5 +- .../Views/SpawnsToolView.axaml.cs | 6 +- Rendering/WDE.MapSpawns/WDE.MapSpawns.csproj | 4 +- Rendering/lib3d.props | 6 +- Updater/Updater.csproj | 2 +- WDE.AzerothCore/WDE.AzerothCore.csproj | 4 +- WDE.Blueprints/Resources/Generic.xaml | 109 -- WDE.Blueprints/Resources/ScrollBarStyle.xaml | 220 --- WDE.Blueprints/Resources/blueprint_icon.png | Bin 2029 -> 0 bytes WDE.Blueprints/WDE.Blueprints.csproj | 8 +- WDE.CMaNGOS/WDE.CMaNGOS.csproj | 4 +- .../LostFocusUpdateBindingBehavior.cs | 16 +- WDE.Common.Avalonia/Components/ButtonImage.cs | 16 +- .../Components/EditableItemsTextBlock.cs | 7 +- .../Components/EditableTextBlock.cs | 6 +- .../MultiViewModelContentPresenter.cs | 113 -- WDE.Common.Avalonia/Components/ToolView.cs | 4 +- .../Components/WdeIconPicker.cs | 6 +- WDE.Common.Avalonia/Components/WdeImage.cs | 209 ++- .../Controls/AsyncDynamicTextBlock.cs | 4 +- .../Controls/AvalonEditExtra.cs | 91 +- .../Controls/BaseGameEnumImage.cs | 64 +- .../Controls/BetterKeyBinding.cs | 30 +- .../Controls/BreakpointIcon.cs | 7 +- .../CodeCompletionService.cs | 45 +- .../CodeCompletionTextEditor.axaml.cs | 12 +- WDE.Common.Avalonia/Controls/ColorPicker.cs | 47 +- .../Controls/CommandTextBlock.cs | 4 +- .../Controls/DialogViewBase.cs | 2 +- .../Controls/DynamicColumnsPanel.cs | 10 +- .../Controls/EditorHeader.axaml | 5 +- .../Controls/EditorHeader.axaml.cs | 7 +- WDE.Common.Avalonia/Controls/FastTreeView.cs | 48 +- .../Controls/FixedContextMenu.cs | 4 +- WDE.Common.Avalonia/Controls/FixedTextBox.cs | 9 +- WDE.Common.Avalonia/Controls/FlagComboBox.cs | 7 +- .../FormattedTextBlock/FormattedTextBlock.cs | 30 +- .../FormattedTextBlock/FormattedTextCache.cs | 16 +- .../FormattedTextBlock/FormattedTextDrawer.cs | 30 +- .../Controls/GameClassesImage.cs | 8 + .../Controls/GameRacesImage.cs | 20 +- WDE.Common.Avalonia/Controls/GameTeamImage.cs | 51 +- .../Controls/HexViewPresenter.cs | 140 +- .../Controls/NullableTextBox.cs | 8 +- .../Controls/PanAndZoom/ZoomBorder.cs | 63 +- .../Controls/ParameterTextBox.cs | 41 +- .../Controls/ParameterValueHolderView.cs | 6 +- .../Controls/Parameters/BaseParameterBox.cs | 212 +++ .../Parameters/CompletionComboParameterBox.cs | 160 ++ .../Controls/Parameters/FlagsParameterBox.cs | 29 + .../Parameters/GenericParameterBox.cs | 188 +++ .../Controls/Parameters/ParameterBox.cs | 177 +++ .../Controls/Parameters/Parameters.axaml | 132 ++ .../Controls/Popups/PopupContextMenu.cs | 6 +- .../Controls/Popups/PopupMenu.axaml | 2 +- .../Controls/Popups/PopupMenu.axaml.cs | 2 +- WDE.Common.Avalonia/Controls/SimpleGraph.cs | 23 +- .../Controls/VirtualizedTreeView.cs | 62 +- .../Controls/VirtualizedTreeViewItem.axaml | 47 +- .../Controls/VirtualizedTreeViewItem.axaml.cs | 2 +- .../Converters/BoolToDoubleConverter.cs | 2 +- .../BoolToRowDefinitionsConverter.cs | 2 +- .../BoolToScrollBarVisibilityConverter.cs | 2 +- .../Converters/BoolToStringConverter.cs | 2 +- .../Converters/DataTimeToStringConverter.cs | 2 +- .../FloatToScaleTransformConverter.cs | 2 +- .../Converters/HueToBrushConverter.cs | 3 +- .../Converters/IntToBoolConverter.cs | 2 +- .../Converters/InverseBoolConverter.cs | 2 +- .../Converters/RgbColorToBrushConverter.cs | 2 +- .../Converters/StripTagConverter.cs | 2 + WDE.Common.Avalonia/DnD/DragAndDrop.cs | 114 +- WDE.Common.Avalonia/Extensions.cs | 14 +- WDE.Common.Avalonia/KeyGestures.cs | 35 +- .../Services/ItemIconDatabase.cs | 2 +- .../Services/SpellIconDatabase.cs | 20 +- .../Services/WebItemIconsService.cs | 4 +- WDE.Common.Avalonia/Themes/Generic.axaml | 1 + .../Themes/ParameterDataTemplateSelector.cs | 4 +- WDE.Common.Avalonia/Types/CompletionData.cs | 3 +- .../Utils/ApplicationExtensions.cs | 24 + WDE.Common.Avalonia/Utils/DataGridColumns.cs | 23 +- .../Utils/DrawingExtensions.cs | 11 +- WDE.Common.Avalonia/Utils/Fix.cs | 4 +- WDE.Common.Avalonia/Utils/FocusUtils.cs | 18 +- WDE.Common.Avalonia/Utils/ICustomCopyPaste.cs | 2 +- WDE.Common.Avalonia/Utils/KeyGestureBind.cs | 4 +- .../Utils/LoadingSpinner.axaml.cs | 2 +- WDE.Common.Avalonia/Utils/MenuBind.cs | 163 +- .../NiceColorGenerator/ColorsGenerator.cs | 1 + WDE.Common.Avalonia/Utils/ResourcesUtils.cs | 6 +- WDE.Common.Avalonia/Utils/Theme.cs | 46 + .../Utils/ThemeAwareSolidBrush.cs | 12 - WDE.Common.Avalonia/Utils/ViewBind.cs | 27 +- .../WDE.Common.Avalonia.csproj | 5 +- WDE.Common.Test/Utils/TaskQueueTests.cs | 10 +- WDE.Common.Test/WDE.Common.Test.csproj | 8 +- .../DbcStore/Views/DBCConfigView.axaml | 4 +- .../DbcStore/Views/DBCConfigView.axaml.cs | 2 +- .../History/Views/HistoryView.axaml | 56 +- .../History/Views/HistoryView.axaml.cs | 7 +- .../{Resources => Icons}/icon_reload.png | Bin .../Mpq/MpqSettingsView.axaml | 2 +- .../Mpq/MpqSettingsView.axaml.cs | 2 +- .../Parameters/SearchConfigurationView.axaml | 4 +- .../SearchConfigurationView.axaml.cs | 2 +- .../Parameters/StringPickerView.axaml.cs | 16 +- .../MultipleParametersPickerView.axaml.cs | 4 +- .../Parameters/Views/ParametersView.axaml | 2 +- .../Parameters/Views/ParametersView.axaml.cs | 2 +- .../Views/UnitBytes1EditorView.axaml.cs | 2 +- .../Views/UnitBytes2EditorView.axaml.cs | 2 +- .../Views/UnixTimestampEditorView.axaml.cs | 2 +- .../RemoteSOAP/Views/SoapConfigView.axaml.cs | 2 +- .../Views/CustomQueryEditorView.axaml.cs | 2 +- .../SQLEditor/Views/SqlEditorToolBar.axaml | 2 +- .../SQLEditor/Views/SqlEditorView.axaml.cs | 2 +- .../Sessions/SessionToolView.axaml | 8 +- .../Sessions/SessionToolView.axaml.cs | 4 +- .../Sessions/SessionsConfigurationView.axaml | 2 +- .../SessionsConfigurationView.axaml.cs | 2 +- .../Explorer/Views/SolutionExplorerView.axaml | 10 +- .../Views/SolutionExplorerView.axaml.cs | 8 +- .../Tools/DebugQueryToolView.axaml | 66 +- .../Tools/DebugQueryToolView.axaml.cs | 10 +- .../Views/DatabaseConfigView.axaml.cs | 2 +- .../Updater/Views/ChangeLogView.axaml | 17 +- .../Updater/Views/ChangeLogView.axaml.cs | 2 +- .../Views/UpdaterConfigurationView.axaml.cs | 2 +- .../WDE.CommonViews.Avalonia.csproj | 4 +- .../Views/ConditionsEditorToolBar.axaml.cs | 2 +- .../Views/ConditionsEditorView.axaml | 26 +- .../Views/ConditionsEditorView.axaml.cs | 2 +- .../WDE.Conditions.Avalonia.csproj | 4 +- .../Data/ConditionDataJsonProvider.cs | 21 +- WDE.Conditions/Data/ConditionDataManager.cs | 18 +- WDE.Conditions/Data/ConditionDataProvider.cs | 46 +- WDE.Conditions/Data/IConditionDataProvider.cs | 12 +- .../SmartData/condition_sources.json | 109 +- .../ConditionObjectEntryParameter.cs | 12 +- WDE.Conditions/WDE.Conditions.csproj | 4 +- .../Controls/FastBoolCellView.axaml.cs | 4 +- .../Controls/FastCellView.axaml | 1 + .../Controls/FastCellView.axaml.cs | 34 +- .../Controls/FastCellViewBase.cs | 57 +- .../Controls/FastFlagCellView.cs | 28 +- .../Controls/FastItemCellView.cs | 28 +- .../Controls/OpenableFastCellViewBase.cs | 26 +- .../Controls/ParameterTextBox.cs | 4 +- .../Helpers/FieldValueTemplateSelector.cs | 4 +- .../Helpers/GridColumnsBinder.cs | 16 +- .../FancyEditorsTablesToolGroupView.axaml | 3 +- .../FancyEditorsTablesToolGroupView.axaml.cs | 17 +- .../Views/ColorsDark.axaml | 2 +- .../Views/ColorsLight.axaml | 2 +- .../Views/MultiRow/CustomCellDrawer.cs | 8 +- .../MultiRowDbTableEditorToolBar.axaml | 12 +- .../MultiRowDbTableEditorToolBar.axaml.cs | 8 +- .../MultiRow/MultiRowDbTableEditorView.axaml | 63 +- .../MultiRowDbTableEditorView.axaml.cs | 6 +- .../Views/MultiRow/SelectablePanel.cs | 98 -- .../OneToOneForeignKeyToolBar.axaml.cs | 2 +- .../OneToOneForeignKeyView.axaml | 26 +- .../OneToOneForeignKeyView.axaml.cs | 2 +- .../Views/RowPickerToolBar.axaml | 6 +- .../Views/RowPickerToolBar.axaml.cs | 2 +- .../Views/RowPickerView.axaml | 8 +- .../Views/RowPickerView.axaml.cs | 2 +- .../Views/SingleRow/CustomCellDrawer.cs | 8 +- .../Views/SingleRow/MySqlFilterControl.axaml | 4 +- .../SingleRow/MySqlFilterControl.axaml.cs | 2 +- .../SingleRowDbTableEditorToolBar.axaml | 12 +- .../SingleRowDbTableEditorToolBar.axaml.cs | 8 +- .../SingleRowDbTableEditorView.axaml | 36 +- .../SingleRowDbTableEditorView.axaml.cs | 6 +- .../Views/Template/HorizontalGridFillPanel.cs | 8 +- .../TemplateDbTableEditorToolBar.axaml | 12 +- .../TemplateDbTableEditorToolBar.axaml.cs | 8 +- .../Template/TemplateDbTableEditorView.axaml | 19 +- .../TemplateDbTableEditorView.axaml.cs | 2 +- .../WDE.DatabaseEditors.Avalonia.csproj | 4 +- .../Data/ContextualParametersJsonProvider.cs | 21 +- .../Data/ContextualParametersProvider.cs | 61 +- .../IContextualParametersJsonProvider.cs | 3 +- .../ITableDefinitionJsonProvider.cs | 6 +- .../Data/Structs/DatabaseColumnJson.cs | 4 +- .../Data/TableDefinitionJsonProvider.cs | 37 +- .../Data/TableDefinitionProvider.cs | 28 +- WDE.DatabaseEditors/DatabaseEditorsModule.cs | 7 +- .../achievement_criteria_data.json | 4 +- ...areatrigger_client_trigger_conditions.json | 47 + .../creature_template_resistance.json | 1 - .../Loaders/DatabaseTableDataProvider.cs | 5 +- .../Parameters/CreatureTextWithFallback.cs | 14 + ...criptRandomTemplateTargetValueParameter.cs | 14 +- .../GossipOptionTextWithFallback.cs | 14 + .../QueryGenerators/HashedEntityPrimaryKey.cs | 1 - .../Services/StandaloneTableEditorSettings.cs | 3 +- .../Services/TableEditorPickerService.cs | 3 +- .../TablesTopBarQuickAccessProvider.cs | 3 + .../DatabaseTableSolutionItemProvider.cs | 6 - .../Solution/QueryReverseRemoteCommand.cs | 5 +- .../ViewModels/CustomObservableCollection.cs | 2 +- .../MultiRow/DatabaseCellViewModel.cs | 3 +- .../MultiRowDbTableEditorViewModel.cs | 25 +- .../SingleRowDbTableEditorViewModel.cs | 8 - .../TemplateDbTableEditorViewModel.cs | 12 - .../WDE.DatabaseEditors.csproj | 13 +- .../Editor/FastGroupingWrapPanel.cs | 84 +- .../UserControls/EventAiEventFlagsView.cs | 8 +- .../Editor/UserControls/EventAiEventView.cs | 8 +- .../Editor/UserControls/EventAiPanelLayout.cs | 97 +- .../Editor/UserControls/EventAiView.axaml | 3 +- .../Editor/UserControls/MiniEventIcon.cs | 2 +- .../SelectableTemplatedControl.cs | 4 +- .../Views/Editing/ParameterEditorView.cs | 6 +- .../Views/Editing/ParametersEditView.axaml | 16 +- .../Editor/Views/EventAiEditorView.axaml.cs | 2 +- .../Editor/Views/EventAiSelectView.axaml | 4 +- .../Editor/Views/EventAiSelectView.axaml.cs | 4 +- .../Views/ParameterDataTemplateSelector.cs | 4 +- .../Editor/Views/ToolEventAiEditorView.axaml | 64 +- .../Themes/Generic.axaml | 16 +- .../WDE.EventAiEditor.Avalonia.csproj | 4 +- WDE.EventAiEditor/Data/EventAiDataManager.cs | 4 - WDE.EventAiEditor/Data/EventAiFactory.cs | 2 +- WDE.EventAiEditor/Editor/IEventAiExporter.cs | 3 +- .../Editing/ParametersEditViewModel.cs | 2 +- .../ViewModels/EventAiEditorViewModel.cs | 9 +- .../ViewModels/EventAiSelectViewModel.cs | 3 +- .../Inspections/IEventActionInspection.cs | 11 - WDE.EventAiEditor/Models/EventAiAction.cs | 5 +- WDE.EventAiEditor/Models/EventAiEvent.cs | 5 +- .../EventAiSetFieldValueParameter.cs | 12 +- .../Providers/EventAiNameProviderBase.cs | 2 +- .../Validation/Antlr/EventAiValidator.cs | 4 +- WDE.EventAiEditor/WDE.EventAiEditor.csproj | 6 +- WDE.MPQ/MpqArchiveSet.cs | 4 +- WDE.MPQ/Services/MpqService.cs | 3 +- WDE.MPQ/ViewModels/MpqSettingsViewModel.cs | 2 +- WDE.MPQ/ViewModels/WoWFilesVerifier.cs | 3 +- WDE.MPQ/WDE.MPQ.csproj | 6 +- .../MVVM/TestObjectNotifyPropertyChanged.cs | 2 +- WDE.MVVM.Test/WDE.MVVM.Test.csproj | 8 +- WDE.MVVM/MVVM/ObservableBase.cs | 27 +- .../Functional/ObservableCollectionStream.cs | 3 +- .../Observable/Functional/SafeObservable.cs | 7 +- WDE.MVVM/Utils/ReferenceEqualityComparer.cs | 2 +- WDE.MVVM/WDE.MVVM.csproj | 11 +- .../Exporter/ExporterHelper.cs | 9 +- .../Exporter/MangosEventAiExporter.cs | 5 +- .../Exporter/MangosEventAiImporter.cs | 1 - .../Providers/EventAiSqlGenerator.cs | 2 +- .../WDE.MangosEventAiEditor.csproj | 6 +- WDE.MapRenderer/GameToolBar.axaml | 74 +- WDE.MapRenderer/GameToolBar.axaml.cs | 2 +- WDE.MapRenderer/GameView.axaml.cs | 8 +- WDE.MapRenderer/GameViewModel.cs | 2 +- WDE.MapRenderer/GameViewService.cs | 2 + WDE.MapRenderer/WDE.MapRenderer.csproj | 6 +- WDE.MpqReader/DBC/Light.cs | 2 - WDE.MpqReader/Structures/BLP.cs | 11 +- WDE.MpqReader/Structures/Structures.cs | 38 +- WDE.MpqReader/WDE.MpqReader.csproj | 6 +- .../CommonModels/MySqlSceneTemplate.cs | 2 +- .../Database/World/CachedDatabaseProvider.cs | 338 ++-- .../Database/World/IAsyncDatabaseProvider.cs | 7 - .../World/NullWorldDatabaseProvider.cs | 124 +- .../Database/World/WorldDatabaseDecorator.cs | 186 ++- .../MySqlDatabaseCommonModule.cs | 10 +- .../Services/CreatureStatCalculatorService.cs | 28 +- .../Services/RollingLogFile.cs | 13 +- .../ViewModels/BaseDatabaseConfigViewModel.cs | 1 - .../WDE.MySqlDatabaseCommon.csproj | 10 +- WDE.PacketViewer.Avalonia/Generic.axaml | 26 + .../MassParsing/MassParserView.axaml | 4 +- .../MassParsing/MassParserView.axaml.cs | 2 +- .../Views/SniffWaypointsDocumentView.axaml | 8 +- .../Views/FilterTextEditor.cs | 4 +- .../Views/PacketDocumentToolBar.axaml | 116 +- .../Views/PacketDocumentToolBar.axaml.cs | 2 +- .../Views/PacketDocumentView.axaml | 104 +- .../Views/PacketDocumentView.axaml.cs | 54 +- .../Views/PacketFilterDialogToolBar.axaml.cs | 2 +- .../Views/PacketFilterDialogView.axaml | 9 +- .../Views/PacketFilterDialogView.axaml.cs | 2 +- .../Views/PacketViewerConfigurationView.axaml | 2 +- .../PacketViewerConfigurationView.axaml.cs | 2 +- .../Views/SearchControl.axaml.cs | 8 +- .../Views/UpdateHistoryView.axaml | 2 +- .../Views/UpdateHistoryView.axaml.cs | 2 +- .../WDE.PacketViewer.Avalonia.csproj | 4 +- .../IntegrationTests/RelatedPacketsTester.cs | 16 +- .../MassParsing/MassParserViewModel.cs | 5 +- .../PacketParserService.cs | 3 +- .../PacketStructures/structures.proto | 1 - .../ActionReaction/RelatedPacketsFinder.cs | 11 +- .../Processors/CreatureTextProcessor.cs | 38 - .../Processors/GossipExtractProcessor.cs | 9 +- .../Processors/GossipMenuProcessor.cs | 35 - .../Processors/UpdateFieldsHistory.cs | 4 +- .../ViewModels/PacketDocumentViewModel.cs | 29 +- .../ViewModels/PacketFilterDialogViewModel.cs | 2 +- .../ViewModels/UpdateHistoryViewModel.cs | 2 +- WDE.PacketViewer/WDE.PacketViewer.csproj | 15 +- .../WDE.Parameters.Test.csproj | 6 +- .../Models/MultiParameterValueHolder.cs | 6 +- .../Models/MultiPropertyValueHolder.cs | 8 +- WDE.Parameters/Models/ParameterValueHolder.cs | 6 +- WDE.Parameters/ParameterFactory.cs | 10 + WDE.Parameters/ParameterLoader.cs | 79 +- .../Providers/IParameterDefinitionProvider.cs | 36 +- .../MultipleParametersPickerViewModel.cs | 2 +- WDE.Parameters/WDE.Parameters.csproj | 16 +- WDE.QuestChainEditor/Resources/icon.png | Bin 2029 -> 0 bytes .../WDE.QuestChainEditor.csproj | 2 +- .../WDE.RemoteSOAP.Test.csproj | 6 +- .../ViewModels/SoapConfigViewModel.cs | 2 +- WDE.RemoteSOAP/WDE.RemoteSOAP.csproj | 4 +- .../Solutions/CustomSqlSolutionItem.cs | 13 +- WDE.SQLEditor/Solutions/Providers.cs | 5 +- .../ViewModels/CustomQueryEditorViewModel.cs | 9 +- .../ViewModels/SqlEditorViewModel.cs | 17 +- WDE.SQLEditor/WDE.SQLEditor.csproj | 3 +- .../Debugging/BreakpointMenuView.axaml | 2 +- .../Debugging/BreakpointMenuView.axaml.cs | 8 +- .../Editor/FastGroupingWrapPanel.cs | 86 +- .../Editor/UserControls/GlobalVariableView.cs | 2 +- .../Editor/UserControls/NoArrangePanel.cs | 17 + .../SelectableTemplatedControl.cs | 8 +- .../Editor/UserControls/SmartConditionView.cs | 2 +- .../UserControls/SmartEventFlagsView.cs | 18 +- .../Editor/UserControls/SmartEventView.cs | 11 +- .../Editor/UserControls/SmartGroupView.cs | 2 +- .../Editor/UserControls/SmartScriptOverlay.cs | 31 +- .../UserControls/SmartScriptPanelLayout.cs | 892 +---------- .../Editor/UserControls/SmartScriptView.axaml | 10 +- .../UserControls/SmartScriptView.axaml.cs | 19 +- .../Editor/UserControls/ToggleAllCheckBox.cs | 11 +- ...VirtualizedSmartScriptPanel.Arrangement.cs | 8 +- ...VirtualizedSmartScriptPanel.Measurement.cs | 8 +- .../VirtualizedSmartScriptPanel.Render.cs | 49 +- .../VirtualizedSmartScriptPanel.cs | 34 +- .../Views/Editing/ParameterEditorView.cs | 6 +- .../Views/Editing/ParametersEditView.axaml | 18 +- .../Views/Editing/ParametersEditView.axaml.cs | 4 +- .../Views/Editing/SmartGroupEditView.axaml.cs | 4 +- .../Views/GlobalVariableEditDialogView.axaml | 2 +- .../GlobalVariableEditDialogView.axaml.cs | 2 +- .../Views/ParameterDataTemplateSelector.cs | 20 +- .../Views/SmartScriptEditorToolBar.axaml | 14 +- .../Views/SmartScriptEditorToolBar.axaml.cs | 8 +- .../Editor/Views/SmartScriptEditorView.axaml | 22 +- .../Views/SmartScriptEditorView.axaml.cs | 2 +- .../Editor/Views/SmartSelectView.axaml | 41 +- .../Editor/Views/SmartSelectView.axaml.cs | 8 +- .../Editor/Views/ToolSmartEditorView.axaml | 58 +- .../Views/VariablePickerToolBar.axaml.cs | 2 +- .../Editor/Views/VariablePickerView.axaml.cs | 2 +- .../Themes/Generic.axaml | 61 +- .../WDE.SmartScriptEditor.Avalonia.csproj | 4 +- .../ViewModels/SmartSelectViewModelTests.cs | 2 +- .../WDE.SmartScriptEditor.Test.csproj | 6 +- .../Data/ISmartDataProvider.cs | 46 +- WDE.SmartScriptEditor/Data/ISmartFactory.cs | 11 +- .../Data/SmartBuiltinRuleParser.cs | 3 +- .../Data/SmartDataManager.cs | 111 +- .../Data/SmartDataProvider.cs | 101 +- WDE.SmartScriptEditor/Data/SmartFactory.cs | 10 +- .../Data/SmartRawDataProvider.cs | 72 +- .../Debugging/ScriptBreakpoints.cs | 3 +- .../Editor/ISmartEditorExtension.cs | 2 +- .../Editor/ISmartScriptDatabaseProvider.cs | 8 +- .../Editor/ISmartScriptExporter.cs | 5 +- .../EditableParameterActionViewModel.cs | 7 + .../Editing/EditableParameterViewModel.cs | 104 +- .../Editing/IEditableParameterViewModel.cs | 4 + .../Editing/ParametersEditViewModel.cs | 2 +- .../ViewModels/SmartScriptEditorViewModel.cs | 19 +- .../Editor/ViewModels/SmartSelectViewModel.cs | 5 +- .../Inspections/AwaitAfterAsyncInspection.cs | 14 +- .../Inspections/IEventActionInspection.cs | 13 - .../Inspections/InspectorService.cs | 22 +- .../Inspections/MustBeLastInspection.cs | 12 +- .../Inspections/NeedsAwaitInspection.cs | 14 +- .../NoWaitInDeathEventInspection.cs | 11 +- .../WaitInCombatEventInspection.cs | 12 +- WDE.SmartScriptEditor/Models/SmartAction.cs | 5 +- WDE.SmartScriptEditor/Models/SmartEvent.cs | 5 +- .../Models/SmartScriptBase.cs | 13 +- WDE.SmartScriptEditor/Models/SmartSource.cs | 5 +- .../Parameters/CreatureSpawnKeyParameter.cs | 29 +- .../Parameters/CreatureTextParameter.cs | 52 +- .../Parameters/GameObjectSpawnKeyParameter.cs | 28 +- .../Parameters/GossipMenuOptionParameter.cs | 20 +- .../NpcFlagsSmartTypeBasedParameter.cs | 12 +- .../Parameters/QuestStarterEnderParameter.cs | 8 +- .../Parameters/SmartScenarioStepParameter.cs | 12 +- .../Parameters/VariableContextualParameter.cs | 24 +- .../Providers/SmartScriptNameProvider.cs | 6 +- .../SmartScriptRelatedProviderBase.cs | 24 +- .../Services/IDynamicContextMenuService.cs | 10 +- .../Services/OutlinerService.cs | 18 +- .../Services/SmartHighlighter.cs | 121 +- .../SmartScriptBaseFindAnywhereSource.cs | 2 +- .../Validation/Antlr/SmartValidator.cs | 2 +- .../WDE.SmartScriptEditor.csproj | 8 +- WDE.Solutions/Resources/folder.png | Bin 437 -> 0 bytes WDE.Solutions/WDE.Solutions.csproj | 13 +- .../SourceCodeConfigurationView.axaml | 2 +- .../SourceCode/FastByteSearcher.cs | 2 +- .../SourceCode/Openers/FileOpenerService.cs | 3 +- .../ViewModels/CommandsDocumentViewModel.cs | 3 +- .../Views/CommandsDocumentView.axaml | 8 +- .../Views/CommandsDocumentView.axaml.cs | 2 +- .../Views/SourceCodeWizardPage.axaml.cs | 2 +- .../Views/TrinityStringsView.axaml | 4 +- .../Views/TrinityStringsView.axaml.cs | 2 +- .../RemoteVisualStudioEnv.cs | 11 +- .../VisualStudioManagerViewModel.cs | 3 +- .../Views/BalloonPopup.axaml.cs | 6 +- .../Views/BreakpointHitPopupView.axaml.cs | 2 +- .../Views/VisualStudioManagerView.axaml | 2 +- .../WDE.SourceCodeIntegrationEditor.csproj | 10 +- .../Parameters/DatabaseSpellParameter.cs | 2 +- WDE.Spells/WDE.Spells.csproj | 4 +- .../WDE.SqlInterpreter.Test.csproj | 10 +- WDE.SqlInterpreter/QueryEvaluator.cs | 3 +- WDE.SqlInterpreter/WDE.SqlInterpreter.csproj | 7 +- WDE.SqlQueryGenerator.Test/DeleteTests.cs | 2 +- .../WDE.SqlQueryGenerator.Test.csproj | 10 +- WDE.SqlQueryGenerator/Extensions.cs | 1 - .../WDE.SqlQueryGenerator.csproj | 6 +- WDE.Trinity/WDE.Trinity.csproj | 4 +- .../Data/SmartDataJsonProvider.cs | 25 +- .../TrinitySmartScriptDatabaseProvider.cs | 11 +- .../Exporter/ExporterHelper.cs | 13 +- .../TrinityCoreSmartScriptExporter.cs | 5 +- .../TrinityCoreSmartScriptImporter.cs | 2 +- .../Providers/SmartScriptSqlGenerator.cs | 4 +- .../SmartScriptSolutionItemProvider.cs | 2 +- .../WDE.TrinitySmartScriptEditor.csproj | 8 +- .../Data/UpdateServerDataProviderTest.cs | 12 +- .../Services/UpdateServiceTest.cs | 2 +- WDE.Updater.Test/WDE.Updater.Test.csproj | 8 +- WDE.Updater/Client/IUpdateClientFactory.cs | 2 +- WDE.Updater/Client/UpdateClient.cs | 6 +- WDE.Updater/Client/UpdateClientFactory.cs | 2 +- WDE.Updater/Client/UpdateVerifier.cs | 2 +- WDE.Updater/Data/IUpdateServerDataProvider.cs | 2 +- WDE.Updater/Data/UpdateServerDataProvider.cs | 8 +- WDE.Updater/Models/CheckVersionRequest.cs | 4 +- .../{Platforms.cs => UpdatePlatforms.cs} | 2 +- .../UpdaterConfigurationViewModel.cs | 2 +- WDE.Updater/WDE.Updater.csproj | 6 +- .../HeadFetchServiceIntegrationTest.cs | 4 +- .../WDE.WoWHeadConnector.Test.csproj | 8 +- WDE.WoWHeadConnector/HeadParser.cs | 3 +- .../WDE.WoWHeadConnector.csproj | 4 +- WDE.WorldMap/FastBoxRendererControl.cs | 11 +- WDE.WorldMap/WDE.WorldMap.csproj | 4 +- WDE.WorldMap/WoWMapRenderer.cs | 7 +- WDE.WorldMap/WoWMapViewer.axaml | 6 +- WDE.WorldMap/WoWMapViewer.axaml.cs | 14 +- WebBuildTasks/ProduceDirectoryListingTask.cs | 22 + WebBuildTasks/WebBuild.Targets | 19 + WebBuildTasks/WebBuildTasks.csproj | 14 + .../Database/BaseDatabaseProvider.cs | 128 +- .../Database/DatabaseProviderClassic.cs | 48 +- .../Database/DatabaseProviderTBC.cs | 48 +- .../Database/DatabaseProviderWoTLK.cs | 48 +- .../Models/AuthDatabase.cs | 4 +- .../Models/Databases/BaseDatabaseTables.cs | 82 +- .../Models/Databases/ClassicDatabase.cs | 24 +- .../Models/Databases/TBCDatabase.cs | 24 +- .../Models/Databases/WoTLKDatabase.cs | 24 +- .../ViewModels/DatabaseConfigViewModel.cs | 2 +- .../WDE.CMMySqlDatabase.csproj | 21 +- .../WorldDatabaseProvider.cs | 4 +- .../Database/ICachedDatabaseProvider.cs | 11 + .../Database/IConversationTemplate.cs | 7 + .../WDE.Common/Database/IDatabaseProvider.cs | 145 +- .../WDE.Common/Database/ISceneTemplate.cs | 2 +- WoWDatabaseEditor.Common/WDE.Common/Log.cs | 182 +++ .../WDE.Common/Managers/IStatusBar.cs | 8 - .../WDE.Common/Managers/IWindowManager.cs | 1 + .../Attributes/AutoRegisterAttribute.cs | 8 + .../Modules/Attributes/Platforms.cs | 19 + .../Modules/IGlobalAsyncInitializer.cs | 10 + .../WDE.Common/Modules/ModuleBase.cs | 32 +- .../WDE.Common/Parameters/IParameter.cs | 66 +- .../Parameters/IParameterFactory.cs | 2 + .../WDE.Common/Services/IClipboardService.cs | 2 +- .../WDE.Common/Services/IGameViewService.cs | 1 + .../Services/IRuntimeDataService.cs | 20 + .../Services/MessageBox/IMessageBoxService.cs | 15 +- .../WDE.Common/Tasks/GlobalApplication.cs | 7 +- .../WDE.Common/Types/ImageUri.cs | 12 +- .../WDE.Common/Utils/AsyncAutoCommand.cs | 23 +- .../WDE.Common/Utils/FlatTreeList.cs | 4 - .../WDE.Common/Utils/TaskExtensions.cs | 53 +- .../WDE.Common/Utils/TaskQueue.cs | 8 +- .../WDE.Common/WDE.Common.csproj | 24 +- .../WDE.DbcStore.Test.csproj | 8 +- .../WDE.DbcStore/DbcStore.cs | 2 - .../WDE.DbcStore/DbcStoreModule.cs | 4 +- .../WDE.DbcStore/FastReader/FastWdc1Reader.cs | 6 +- .../WDE.DbcStore/WDE.DbcStore.csproj | 20 +- .../WDE.History/HistoryManager.cs | 6 +- .../ViewModels/HistoryViewModel.cs | 10 +- .../WDE.History/WDE.History.csproj | 3 +- .../Database/AzerothhMySqlDatabaseProvider.cs | 54 +- .../BaseTrinityMySqlDatabaseProvider.cs | 157 +- .../TrinityCataMySqlDatabaseProvider.cs | 64 +- .../TrinityMasterMySqlDatabaseProvider.cs | 75 +- .../TrinityWrathMySqlDatabaseProvider.cs | 54 +- .../Models/Databases/AzerothDatabase.cs | 32 +- .../Models/Databases/BaseTrinityDatabase.cs | 82 +- .../Models/Databases/TrinityCataDatabase.cs | 60 +- .../Models/Databases/TrinityMasterDatabase.cs | 44 +- .../Models/Databases/TrinityWrathDatabase.cs | 34 +- .../Models/TrinityAuthDatabase.cs | 4 +- .../WDE.TrinityMySqlDatabase.csproj | 21 +- WoWDatabaseEditor.sln | 73 +- .../CoreVersion/CurrentCoreVersionService.cs | 2 +- WoWDatabaseEditor/Icons/document_log.png | Bin 0 -> 961 bytes WoWDatabaseEditor/Icons/document_log@2x.png | Bin 0 -> 1436 bytes WoWDatabaseEditor/Icons/document_log_big.png | Bin 0 -> 1436 bytes .../Icons/document_log_big@2x.png | Bin 0 -> 3066 bytes WoWDatabaseEditor/Managers/DocumentManager.cs | 2 +- .../Managers/FatalErrorHandler.cs | 3 +- .../Managers/SessionRestoreService.cs | 6 +- WoWDatabaseEditor/Managers/StatusBar.cs | 13 +- WoWDatabaseEditor/Managers/ViewLocator.cs | 4 +- .../ViewModels/ModulesConfigViewModel.cs | 2 + .../Providers/HelpMenuItemProvider.cs | 5 +- .../ViewModels/ConfigurationPanelViewModel.cs | 2 +- .../Services/DebugConsole/DebugConsole.cs | 4 +- .../Services/DotNetUtils/IDotNetService.cs | 3 +- .../Services/FileSystemService/FileSystem.cs | 3 +- .../FindAnywhereResultsViewModel.cs | 9 +- .../Services/Http/UrlOpenService.cs | 3 +- .../LoadingEvents/LoadingEventAggregator.cs | 2 - .../LogService/Extensions/LoggerExtensions.cs | 56 + .../LogService/Logging/DataStoreLoggerSink.cs | 85 + .../Logging/DataStoreLoggerSinkExtensions.cs | 16 + .../LogService/Logging/ILogDataStore.cs | 11 + .../LogService/Logging/ILogDataStoreImpl.cs | 6 + .../LogService/Logging/LogDataStore.cs | 16 + .../Services/LogService/Logging/LogModel.cs | 21 + .../IReportErrorsConfigService.cs | 9 + .../ReportErrorsConfigService.cs | 37 + .../ReportErrorsConfigurationViewModel.cs | 33 + .../ReportErrorsToServer/ReportErrorsSink.cs | 100 ++ .../ReportErrorsSinkManager.cs | 69 + .../ViewModels/LogViewerControlViewModel.cs | 43 + .../MostRecentlyUsedService.cs | 2 +- .../PersonalGuidConfigurationViewModel.cs | 2 +- .../RuntimeData/LocalRuntimeDataService.cs | 57 + .../RuntimeData/WebRuntimeDataService.cs | 91 ++ .../ServerExecutableConfigurationPanel.cs | 2 +- .../ServerIntegration/ServerIntegration.cs | 7 +- .../TextDocumentService.cs | 12 +- .../UserSettingsService/UserSettings.cs | 6 +- .../UserSettingsService/WebUserSettings.cs | 65 + WoWDatabaseEditor/Tasks/TaskRunner.cs | 13 +- .../ViewModels/AboutViewModel.cs | 4 +- .../ViewModels/MainWindowViewModel.cs | 14 +- .../ViewModels/QuickStartViewModel.cs | 29 +- .../ViewModels/TasksViewModel.cs | 3 +- .../WoWDatabaseEditorCore.csproj | 14 +- WoWDatabaseEditorCore.Avalonia/App.xaml | 135 +- WoWDatabaseEditorCore.Avalonia/App.xaml.cs | 234 ++- .../AvaloniaViewLocator.cs | 2 +- .../LogViewer/Converters/EventIdConverter.cs | 23 + .../LogLevelToForegroundConverter.cs | 70 + .../LogViewer/LogViewer.Avalonia.csproj | 30 + .../Controls/LogViewer/LogViewerControl.axaml | 60 + .../LogViewer/LogViewerControl.axaml.cs | 47 + .../Controls/SingleView/FakeWindow.axaml | 34 + .../Controls/SingleView/FakeWindow.axaml.cs | 112 ++ .../SingleView/FakeWindowControl.axaml | 30 + .../SingleView/FakeWindowControl.axaml.cs | 9 + .../Controls/SingleView/PseudoWindowsPanel.cs | 42 + .../Controls/SingleView/WindowPanel.cs | 64 + .../Views/CoreVersionConfigView.axaml | 2 +- .../Views/CoreVersionConfigView.axaml.cs | 2 +- .../CustomModuleInitializer.cs | 1 + .../Docking/AvaloniaDockAdapter.cs | 19 +- .../Docking/AvaloniaDocumentDockWrapper.cs | 2 +- .../Docking/AvaloniaToolDockWrapper.cs | 4 +- .../Docking/DockFactory.cs | 11 +- .../Docking/FocusAwareDocumentDock.cs | 2 +- .../Docking/PersistentDockDataTemplate.cs | 102 +- .../Docking/Serialization/DockSerializer.cs | 108 +- .../Docking/Serialization/SerializedDock.cs | 1 + .../Extensions/WindowExtensions.cs | 4 +- .../Fonts/unifont.otf | Bin 5105244 -> 0 bytes .../MainModuleAvalonia.cs | 3 + .../Managers/MessageBoxService.cs | 100 +- .../Managers/ScopedContainer.cs | 1 - .../Managers/WindowManager.cs | 322 ++-- .../Views/ModulesConfigView.axaml.cs | 2 +- WoWDatabaseEditorCore.Avalonia/Program.cs | 141 +- .../AppearanceService/AvaloniaThemeStyle.cs | 11 +- .../ViewModels/ThemeConfigViewModel.cs | 4 +- .../Views/ThemeConfigView.axaml | 8 +- .../Services/ClipboardService.cs | 12 +- .../Views/ConfigurationPanelView.axaml | 10 +- .../Views/ConfigurationPanelView.axaml.cs | 2 +- .../GenericSelectorDialogView.axaml.cs | 14 +- .../CreatureEntryProviderService.cs | 14 +- .../GameObjectEntryProviderService.cs | 14 +- .../QuestEntryProviderService.cs | 13 +- .../FindAnywhere/FindAnywhereDialogView.axaml | 2 +- .../FindAnywhereDialogView.axaml.cs | 2 +- .../FindAnywhereResultsView.axaml | 2 +- .../FindAnywhereResultsView.axaml.cs | 2 +- .../GenericSettings/GeneralSettingsView.axaml | 22 +- .../GeneralSettingsView.axaml.cs | 2 +- .../InputEntryProviderView.axaml.cs | 2 +- .../ItemFromListProviderView.xaml | 1 - .../ItemFromListProviderView.xaml.cs | 13 +- .../MemoryUsageDebugToolView.axaml.cs | 2 +- .../MessageBoxControlView.axaml | 54 + .../MessageBoxControlView.axaml.cs | 18 + .../MessageBoxService/MessageBoxView.axaml | 13 +- .../MessageBoxService/MessageBoxView.axaml.cs | 84 +- .../MessageBoxService/MessageBoxViewModel.cs | 17 +- .../NewItemService/NewItemDialogView.axaml | 8 +- .../NewItemService/NewItemDialogView.axaml.cs | 4 +- .../OutlinerTool/OutlinerFastTreeView.cs | 1 + .../OutlinerTool/OutlinerToolView.axaml | 2 +- .../OutlinerTool/OutlinerToolView.axaml.cs | 8 +- .../PersonalGuidConfigurationView.axaml.cs | 2 +- .../Services/ProblemsTool/ProblemsView.axaml | 15 +- .../ProblemsTool/ProblemsView.axaml.cs | 2 +- .../Services/Profiles/ProfileCreateView.axaml | 2 +- .../Profiles/ProfileCreateView.axaml.cs | 2 +- .../Services/Profiles/ProfilesControl.axaml | 26 +- .../Profiles/ProfilesControl.axaml.cs | 2 +- .../QuickAccess/QuickAccessView.axaml | 2 +- .../QuickAccess/QuickAccessView.axaml.cs | 10 +- .../QuickLoadSettingsView.axaml | 4 +- .../QuickLoadSettingsView.axaml.cs | 2 +- ...rExecutableConfigurationPanelView.axaml.cs | 2 +- .../TabularDataPickerView.axaml.cs | 11 +- .../TabularDataPickerViewModel.cs | 10 +- .../Views/AboutView.axaml.cs | 2 +- .../Views/DebugConsoleToolBar.axaml.cs | 2 +- .../Views/DebugConsoleView.axaml.cs | 2 +- .../Views/DialogWindow.axaml | 5 +- .../Views/DialogWindow.axaml.cs | 87 +- .../Views/IMainWindowHolder.cs | 20 +- .../Views/MainWebView.axaml | 273 ++++ .../Views/MainWebView.axaml.cs | 60 + .../Views/MainWindow.xaml | 157 -- .../Views/MainWindow.xaml.cs | 104 -- .../Views/MainWindowWithDocking.xaml | 54 +- .../Views/MainWindowWithDocking.xaml.cs | 30 +- .../Views/QuickStartView.axaml | 51 +- .../Views/QuickStartView.axaml.cs | 55 +- .../Views/RemoteConnectionToolView.axaml.cs | 2 - .../Views/SplashScreenWindow.axaml.cs | 24 +- .../Views/StatusBarView.axaml | 8 +- .../Views/StatusBarView.axaml.cs | 2 +- .../Views/TablesToolView.axaml | 2 +- .../Views/TasksPanel.axaml | 24 +- .../Views/TasksPanel.axaml.cs | 2 +- .../Views/TextDocumentToolBar.axaml.cs | 2 +- .../Views/TextDocumentView.axaml.cs | 2 +- .../Views/TopBarQuickAccessView.axaml | 2 +- .../Views/WdeToolsTabControl.cs | 19 - .../Views/Window1.axaml | 9 + .../Views/Window1.axaml.cs | 25 + .../Views/WindowToolbarDocumentWrapper.axaml | 2 +- .../Views/WindowViewDocumentWrapper.axaml | 2 +- .../Views/WrapPanelWithDivider.cs | 10 +- .../WoWDatabaseEditorCore.Avalonia.csproj | 20 +- .../AppVersion/AppDataProviderTest.cs | 2 +- .../AppVersion/ApplicationVersionTest.cs | 6 +- .../VirtualFileSystemTest.cs | 2 +- .../UserSettingsService/UserSettingsTest.cs | 6 +- .../WoWDatabaseEditorCore.Test.csproj | 6 +- WoWPacketParser | 2 +- .../WoWPacketParserLoader.csproj | 2 +- appveyor.yml | 111 +- build.sh | 113 ++ global.json | 2 +- 1127 files changed, 19965 insertions(+), 10866 deletions(-) create mode 100644 AvaloniaStyles.Desktop/AvaloniaStyles.Desktop.csproj create mode 100644 AvaloniaStyles.Desktop/Program.cs delete mode 100644 AvaloniaStyles/Controls/AlternativeScrollViewer.cs create mode 100644 AvaloniaStyles/Controls/ControlRenderer.cs delete mode 100644 AvaloniaStyles/Controls/DropDownButton.axaml delete mode 100644 AvaloniaStyles/Controls/DropDownButton.axaml.cs create mode 100644 AvaloniaStyles/Controls/HamburgerMenuButton.axaml create mode 100644 AvaloniaStyles/Controls/HamburgerMenuButton.axaml.cs delete mode 100644 AvaloniaStyles/Controls/LightSolidColorBrush.cs delete mode 100644 AvaloniaStyles/Controls/ManagedMenu.cs delete mode 100644 AvaloniaStyles/Controls/ToolbarControl.axaml.cs create mode 100644 AvaloniaStyles/Demo/DemoDataContext.cs create mode 100644 AvaloniaStyles/Demo/MainView.axaml create mode 100644 AvaloniaStyles/Demo/MainView.axaml.cs rename AvaloniaStyles/Styles/{ => MacOs}/MacOsBigSurDark.xaml (100%) rename AvaloniaStyles/Styles/{ => MacOs}/MacOsBigSurLight.xaml (100%) rename AvaloniaStyles/Styles/{ => MacOs}/MacOsCatalinaDark.xaml (100%) rename AvaloniaStyles/Styles/{ => MacOs}/MacOsCatalinaLight.xaml (100%) create mode 100644 AvaloniaStyles/Styles/Windows10/ColorsDark.axaml delete mode 100644 AvaloniaStyles/Styles/Windows10/ColorsDark.xaml rename AvaloniaStyles/Styles/Windows10/{ColorsLight.xaml => ColorsLight.axaml} (61%) create mode 100644 AvaloniaStyles/Styles/Windows10/Dock.axaml rename AvaloniaStyles/Styles/Windows10/{ListBox.xaml => ListBox.axaml} (100%) delete mode 100644 AvaloniaStyles/Styles/Windows10/NativeMenuBar.xaml rename AvaloniaStyles/Styles/Windows10/{SideBar.xaml => SideBar.axaml} (100%) rename AvaloniaStyles/Styles/Windows10/{StatusBar.xaml => StatusBar.axaml} (88%) rename AvaloniaStyles/Styles/Windows10/{Style.xaml => Style.axaml} (70%) rename AvaloniaStyles/Styles/Windows10/{TabStrip.xaml => TabStrip.axaml} (96%) delete mode 100644 AvaloniaStyles/Styles/Windows10/ToolsTabControl.xaml create mode 100644 AvaloniaStyles/Styles/Windows10/Window.axaml delete mode 100644 AvaloniaStyles/Styles/Windows10/Window.xaml create mode 100644 AvaloniaStyles/Styles/Windows10/WindowMessageBox.axaml rename AvaloniaStyles/Styles/{Windows10Dark.xaml => Windows10Dark.axaml} (74%) rename AvaloniaStyles/Styles/{Windows10Light.xaml => Windows10Light.axaml} (72%) create mode 100644 AvaloniaStyles/Styles/Windows10Light.cs rename AvaloniaStyles/Styles/Windows11/{ColorsDark.xaml => ColorsDark.axaml} (62%) rename AvaloniaStyles/Styles/Windows11/{ColorsLight.xaml => ColorsLight.axaml} (61%) delete mode 100644 AvaloniaStyles/Styles/Windows11/NativeMenuBar.xaml rename AvaloniaStyles/Styles/Windows11/{Style.xaml => Style.axaml} (71%) rename AvaloniaStyles/Styles/{Windows11Dark.xaml => Windows11Dark.axaml} (74%) rename AvaloniaStyles/Styles/{Windows11Light.xaml => Windows11Light.axaml} (79%) delete mode 100644 AvaloniaStyles/Utils/SolidBrushTransition.cs create mode 100644 BaseDesktopLoader/BaseDesktopLoader.csproj create mode 100644 BaseDesktopLoader/Program.cs create mode 100644 LoaderAvalonia.Web/AppBundle/Logo.svg create mode 100644 LoaderAvalonia.Web/AppBundle/app.css create mode 100644 LoaderAvalonia.Web/AppBundle/favicon.ico create mode 100644 LoaderAvalonia.Web/AppBundle/index.html create mode 100644 LoaderAvalonia.Web/AppBundle/main.js create mode 100644 LoaderAvalonia.Web/LoaderAvalonia.Web.csproj create mode 100644 LoaderAvalonia.Web/NullDebuggerService.cs create mode 100644 LoaderAvalonia.Web/NullGameView.cs create mode 100644 LoaderAvalonia.Web/NullRemoteConnectorService.cs create mode 100644 LoaderAvalonia.Web/NullSourceCodePathService.cs create mode 100644 LoaderAvalonia.Web/NullUpdateViewModel.cs create mode 100644 LoaderAvalonia.Web/NullVisualStudioManagerViewModel.cs create mode 100644 LoaderAvalonia.Web/Program.cs create mode 100644 LoaderAvalonia.Web/Properties/launchSettings.json create mode 100644 LoaderAvalonia.Web/WebModule.cs create mode 100644 LoaderAvalonia.Web/runtimeconfig.template.json create mode 100644 LoaderAvalonia.iOS/AppDelegate.cs create mode 100644 LoaderAvalonia.iOS/Entitlements.plist create mode 100644 LoaderAvalonia.iOS/Info.plist create mode 100644 LoaderAvalonia.iOS/LoaderAvalonia.iOS.csproj create mode 100644 LoaderAvalonia.iOS/Main.cs create mode 100644 LoaderAvalonia.iOS/NullDebuggerService.cs create mode 100644 LoaderAvalonia.iOS/NullGameView.cs create mode 100644 LoaderAvalonia.iOS/NullRemoteConnectorService.cs create mode 100644 LoaderAvalonia.iOS/NullSourceCodePathService.cs create mode 100644 LoaderAvalonia.iOS/NullUpdateViewModel.cs create mode 100644 LoaderAvalonia.iOS/NullVisualStudioManagerViewModel.cs create mode 100644 LoaderAvalonia.iOS/Resources/LaunchScreen.xib create mode 100644 LoaderAvalonia.iOS/iOSModule.cs create mode 100644 Modules/WDE.HttpDatabase/HttpDatabaseModule.cs create mode 100644 Modules/WDE.HttpDatabase/HttpDatabaseProvider.cs create mode 100644 Modules/WDE.HttpDatabase/HttpDatabaseProviderImpl.cs create mode 100644 Modules/WDE.HttpDatabase/Mock/MockHotfixDatabaseSettingsProvider.cs create mode 100644 Modules/WDE.HttpDatabase/Models/AzerothMySqlSmartScriptLine.cs create mode 100644 Modules/WDE.HttpDatabase/Models/AzerothMySqlSpellDbc.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonAreaTriggerCreateProperties.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonAzerothString.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonConversationActor.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonConversationActorTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonCoreCommandHelp.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonCreatureTemplateAndDifficulty.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonLootTemplateName.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonNpcSpellClickSpell.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonSpawnGroupFormation.cs create mode 100644 Modules/WDE.HttpDatabase/Models/JsonTrinityString.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlAreaTriggerScript.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlAreaTriggerTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlBroadcastText.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlBroadcastTextLocale.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlConditionLine.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlConversationTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureAddon.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureClassLevelStat.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureEquipmentTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureModelInfo.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureTemplateDifficulty.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureTemplateWrath.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureText.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlCreatureWrath.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlEventScriptBaseLine.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGameEvent.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGameEventCreature.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGameEventGameObject.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGameObjectTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGameObjectWrath.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGossipMenu.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGossipMenuLine.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlGossipMenuOptionWrath.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlItemTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlLootEntry.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlNpcText.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlPhaseName.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlPlayerChoice.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlPointOfInterest.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlQuestObjective.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlQuestRequestItem.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlQuestTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlQuestTemplateAddon.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlSceneTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlSmartScriptLine.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlSpawnGroupSpawn.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlSpawnGroupTemplate.cs create mode 100644 Modules/WDE.HttpDatabase/Models/MySqlSpellScriptName.cs create mode 100644 Modules/WDE.HttpDatabase/Models/RbacLinkedPermission.cs create mode 100644 Modules/WDE.HttpDatabase/Models/RbacPermission.cs create mode 100644 Modules/WDE.HttpDatabase/Models/TrinityMySqlSpellDbc.cs create mode 100644 Modules/WDE.HttpDatabase/Models/Waypoints.cs create mode 100644 Modules/WDE.HttpDatabase/QueryExecutor/DummyAuthMySqlExecutor.cs create mode 100644 Modules/WDE.HttpDatabase/QueryExecutor/DummyHotfixMySqlExecutor.cs create mode 100644 Modules/WDE.HttpDatabase/QueryExecutor/HttpMySqlExecutor.cs create mode 100644 Modules/WDE.HttpDatabase/QueryExecutor/MySqlType.cs create mode 100644 Modules/WDE.HttpDatabase/QueryExecutor/SelectResult.cs create mode 100644 Modules/WDE.HttpDatabase/WDE.HttpDatabase.csproj create mode 100644 Rendering/RenderingTester/OpenTKDevice.cs rename Rendering/{TheEngine => RenderingTester}/TheEngineOpenTkWindow.cs (99%) delete mode 100644 WDE.Blueprints/Resources/Generic.xaml delete mode 100644 WDE.Blueprints/Resources/ScrollBarStyle.xaml delete mode 100644 WDE.Blueprints/Resources/blueprint_icon.png delete mode 100644 WDE.Common.Avalonia/Components/MultiViewModelContentPresenter.cs create mode 100644 WDE.Common.Avalonia/Controls/Parameters/BaseParameterBox.cs create mode 100644 WDE.Common.Avalonia/Controls/Parameters/CompletionComboParameterBox.cs create mode 100644 WDE.Common.Avalonia/Controls/Parameters/FlagsParameterBox.cs create mode 100644 WDE.Common.Avalonia/Controls/Parameters/GenericParameterBox.cs create mode 100644 WDE.Common.Avalonia/Controls/Parameters/ParameterBox.cs create mode 100644 WDE.Common.Avalonia/Controls/Parameters/Parameters.axaml create mode 100644 WDE.Common.Avalonia/Utils/ApplicationExtensions.cs create mode 100644 WDE.Common.Avalonia/Utils/Theme.cs delete mode 100644 WDE.Common.Avalonia/Utils/ThemeAwareSolidBrush.cs rename WDE.CommonViews.Avalonia/{Resources => Icons}/icon_reload.png (100%) delete mode 100644 WDE.DatabaseEditors.Avalonia/Views/MultiRow/SelectablePanel.cs create mode 100644 WDE.DatabaseEditors/DbDefinitions/areatrigger_client_trigger_conditions.json create mode 100644 WDE.PacketViewer.Avalonia/Generic.axaml delete mode 100644 WDE.PacketViewer/Processing/Processors/CreatureTextProcessor.cs delete mode 100644 WDE.PacketViewer/Processing/Processors/GossipMenuProcessor.cs delete mode 100644 WDE.QuestChainEditor/Resources/icon.png create mode 100644 WDE.SmartScriptEditor.Avalonia/Editor/UserControls/NoArrangePanel.cs delete mode 100644 WDE.Solutions/Resources/folder.png rename WDE.Updater/Models/{Platforms.cs => UpdatePlatforms.cs} (84%) create mode 100644 WebBuildTasks/ProduceDirectoryListingTask.cs create mode 100644 WebBuildTasks/WebBuild.Targets create mode 100644 WebBuildTasks/WebBuildTasks.csproj create mode 100644 WoWDatabaseEditor.Common/WDE.Common/Log.cs create mode 100644 WoWDatabaseEditor.Common/WDE.Common/Modules/Attributes/Platforms.cs create mode 100644 WoWDatabaseEditor.Common/WDE.Common/Modules/IGlobalAsyncInitializer.cs create mode 100644 WoWDatabaseEditor.Common/WDE.Common/Services/IRuntimeDataService.cs create mode 100644 WoWDatabaseEditor/Icons/document_log.png create mode 100644 WoWDatabaseEditor/Icons/document_log@2x.png create mode 100644 WoWDatabaseEditor/Icons/document_log_big.png create mode 100644 WoWDatabaseEditor/Icons/document_log_big@2x.png create mode 100644 WoWDatabaseEditor/Services/LogService/Extensions/LoggerExtensions.cs create mode 100644 WoWDatabaseEditor/Services/LogService/Logging/DataStoreLoggerSink.cs create mode 100644 WoWDatabaseEditor/Services/LogService/Logging/DataStoreLoggerSinkExtensions.cs create mode 100644 WoWDatabaseEditor/Services/LogService/Logging/ILogDataStore.cs create mode 100644 WoWDatabaseEditor/Services/LogService/Logging/ILogDataStoreImpl.cs create mode 100644 WoWDatabaseEditor/Services/LogService/Logging/LogDataStore.cs create mode 100644 WoWDatabaseEditor/Services/LogService/Logging/LogModel.cs create mode 100644 WoWDatabaseEditor/Services/LogService/ReportErrorsToServer/IReportErrorsConfigService.cs create mode 100644 WoWDatabaseEditor/Services/LogService/ReportErrorsToServer/ReportErrorsConfigService.cs create mode 100644 WoWDatabaseEditor/Services/LogService/ReportErrorsToServer/ReportErrorsConfigurationViewModel.cs create mode 100644 WoWDatabaseEditor/Services/LogService/ReportErrorsToServer/ReportErrorsSink.cs create mode 100644 WoWDatabaseEditor/Services/LogService/ReportErrorsToServer/ReportErrorsSinkManager.cs create mode 100644 WoWDatabaseEditor/Services/LogService/ViewModels/LogViewerControlViewModel.cs create mode 100644 WoWDatabaseEditor/Services/RuntimeData/LocalRuntimeDataService.cs create mode 100644 WoWDatabaseEditor/Services/RuntimeData/WebRuntimeDataService.cs create mode 100644 WoWDatabaseEditor/Services/UserSettingsService/WebUserSettings.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/LogViewer/Converters/EventIdConverter.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/LogViewer/Converters/LogLevelToForegroundConverter.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/LogViewer/LogViewer.Avalonia.csproj create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/LogViewer/LogViewerControl.axaml create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/LogViewer/LogViewerControl.axaml.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/SingleView/FakeWindow.axaml create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/SingleView/FakeWindow.axaml.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/SingleView/FakeWindowControl.axaml create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/SingleView/FakeWindowControl.axaml.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/SingleView/PseudoWindowsPanel.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Controls/SingleView/WindowPanel.cs delete mode 100644 WoWDatabaseEditorCore.Avalonia/Fonts/unifont.otf create mode 100644 WoWDatabaseEditorCore.Avalonia/Services/MessageBoxService/MessageBoxControlView.axaml create mode 100644 WoWDatabaseEditorCore.Avalonia/Services/MessageBoxService/MessageBoxControlView.axaml.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Services/OutlinerTool/OutlinerFastTreeView.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Views/MainWebView.axaml create mode 100644 WoWDatabaseEditorCore.Avalonia/Views/MainWebView.axaml.cs delete mode 100644 WoWDatabaseEditorCore.Avalonia/Views/MainWindow.xaml delete mode 100644 WoWDatabaseEditorCore.Avalonia/Views/MainWindow.xaml.cs delete mode 100644 WoWDatabaseEditorCore.Avalonia/Views/WdeToolsTabControl.cs create mode 100644 WoWDatabaseEditorCore.Avalonia/Views/Window1.axaml create mode 100644 WoWDatabaseEditorCore.Avalonia/Views/Window1.axaml.cs create mode 100644 build.sh diff --git a/AntlrSupport.props b/AntlrSupport.props index 57a5f2e00..27824047b 100644 --- a/AntlrSupport.props +++ b/AntlrSupport.props @@ -1,10 +1,6 @@ - - $(MSBuildProjectDirectory)/obj - $(BaseIntermediateOutputPath)/$(Configuration) + \ No newline at end of file diff --git a/Avalonia.props b/Avalonia.props index 6251618c4..7e6c37e01 100644 --- a/Avalonia.props +++ b/Avalonia.props @@ -1,12 +1,18 @@ + + 11.1.999-cibuild0045870-beta + + + + - - - - + + + + @@ -17,4 +23,4 @@ Designer - \ No newline at end of file + diff --git a/AvaloniaEdit b/AvaloniaEdit index 0f29494c1..e157a49ea 160000 --- a/AvaloniaEdit +++ b/AvaloniaEdit @@ -1 +1 @@ -Subproject commit 0f29494c1446c91e43ba58e6dbd02837a1183d79 +Subproject commit e157a49ea4b8185c4f38597366562f9ee88c30c6 diff --git a/AvaloniaStyles.Desktop/AvaloniaStyles.Desktop.csproj b/AvaloniaStyles.Desktop/AvaloniaStyles.Desktop.csproj new file mode 100644 index 000000000..e00ef64e0 --- /dev/null +++ b/AvaloniaStyles.Desktop/AvaloniaStyles.Desktop.csproj @@ -0,0 +1,22 @@ + + + + WinExe + net8.0 + Debug;Release + AnyCPU + enable + $(WarningsAsErrors),nullable + + + + + + + + + + + + + diff --git a/AvaloniaStyles.Desktop/Program.cs b/AvaloniaStyles.Desktop/Program.cs new file mode 100644 index 000000000..6b967034b --- /dev/null +++ b/AvaloniaStyles.Desktop/Program.cs @@ -0,0 +1,16 @@ +using Avalonia; + +namespace AvaloniaStyles.Desktop; + +public class Program +{ + static void Main(string[] args) + => BuildAvaloniaApp().StartWithClassicDesktopLifetime(args); + + // App configuration, used by the entry point and previewer + static AppBuilder BuildAvaloniaApp() + => AppBuilder.Configure() + .UsePlatformDetect() + .WithInterFont() + .LogToTrace(); +} \ No newline at end of file diff --git a/AvaloniaStyles/App.xaml b/AvaloniaStyles/App.xaml index 22f278679..fbafe1a4b 100644 --- a/AvaloniaStyles/App.xaml +++ b/AvaloniaStyles/App.xaml @@ -5,4 +5,8 @@ + + + + diff --git a/AvaloniaStyles/App.xaml.cs b/AvaloniaStyles/App.xaml.cs index 70c650d51..d1d4ebd00 100644 --- a/AvaloniaStyles/App.xaml.cs +++ b/AvaloniaStyles/App.xaml.cs @@ -16,20 +16,14 @@ public override void OnFrameworkInitializationCompleted() { if (ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop) desktop.MainWindow = new MainWindow(); + else if (ApplicationLifetime is ISingleViewApplicationLifetime singleViewPlatform) + { + singleViewPlatform.MainView = new MainView + { + DataContext = new DemoDataContext() + }; + } base.OnFrameworkInitializationCompleted(); } - - static void Main(string[] args) - => BuildAvaloniaApp().StartWithClassicDesktopLifetime(args); - - // App configuration, used by the entry point and previewer - static AppBuilder BuildAvaloniaApp() - => AppBuilder.Configure() - .With(new Win32PlatformOptions - { - OverlayPopups = true, - }) - .UsePlatformDetect() - .LogToTrace(); } } \ No newline at end of file diff --git a/AvaloniaStyles/AvaloniaStyles.csproj b/AvaloniaStyles/AvaloniaStyles.csproj index a9b86af7d..087a49a65 100644 --- a/AvaloniaStyles/AvaloniaStyles.csproj +++ b/AvaloniaStyles/AvaloniaStyles.csproj @@ -1,19 +1,17 @@ - WinExe - net7.0 - win8-x64;osx-x64;linux-x64;osx-arm64;win-x64 + Library + net8.0 Debug;Release AnyCPU enable - nullable + $(WarningsAsErrors),nullable - false - true - ..\bin\$(Configuration)\ + true + @@ -26,6 +24,9 @@ + + + diff --git a/AvaloniaStyles/Controls/AccentSolidColorBrush.cs b/AvaloniaStyles/Controls/AccentSolidColorBrush.cs index ba0e7345a..c26caf120 100644 --- a/AvaloniaStyles/Controls/AccentSolidColorBrush.cs +++ b/AvaloniaStyles/Controls/AccentSolidColorBrush.cs @@ -1,86 +1,55 @@ using Avalonia; using Avalonia.Media; -using Avalonia.Media.Immutable; using Avalonia.Metadata; using AvaloniaStyles.Utils; +using HslColor = AvaloniaStyles.Utils.HslColor; namespace AvaloniaStyles.Controls; -public class AccentSolidColorBrush : Brush, ISolidColorBrush +public class Accent { - public static readonly StyledProperty BaseColorProperty = - AvaloniaProperty.Register(nameof(BaseColor)); - - public static readonly StyledProperty HueProperty = - AvaloniaProperty.Register(nameof(Hue)); - - static AccentSolidColorBrush() - { - AffectsRender(BaseColorProperty); - AffectsRender(HueProperty); - } - - public AccentSolidColorBrush(Color color, double opacity = 1.0) - { - BaseColor = color; - Opacity = opacity; - } - - public AccentSolidColorBrush(uint color) : this(Color.FromUInt32(color)) - { - } - - public AccentSolidColorBrush() - { - Opacity = 1; - } - - /// Gets or sets the color of the brush. - [Content] - public Color BaseColor - { - get => GetValue(BaseColorProperty); - set => SetValue(BaseColorProperty, value); - } - - public HslDiff Hue - { - get => GetValue(HueProperty); - set => SetValue(HueProperty, value); - } - - /// Parses a brush string. - /// The brush string. - /// The . - /// - /// Whereas may return an immutable solid color brush, - /// this method always returns a mutable . - /// - public static AccentSolidColorBrush Parse(string s) - { - ISolidColorBrush solidColorBrush1 = (ISolidColorBrush)Brush.Parse(s); - return !(solidColorBrush1 is AccentSolidColorBrush solidColorBrush2) - ? new AccentSolidColorBrush(solidColorBrush1.Color) - : solidColorBrush2; - } - - public override string ToString() => Color.ToString(); - - /// - public override IBrush ToImmutable() => new ImmutableSolidColorBrush(this); - - public Color Color - { - get - { - var hsl = HslColor.FromRgba(BaseColor).Scale(Hue); - return hsl.ToRgba(); - } - set - { - var hsl = HslColor.FromRgba(value); - //Hue = (float)hsl.H; - BaseColor = value; - } - } + public static readonly StyledProperty BaseColorProperty = + AvaloniaProperty.RegisterAttached("BaseColor"); + + public static readonly StyledProperty HueProperty = + AvaloniaProperty.RegisterAttached("Hue"); + + static Accent() + { + HueProperty.Changed.AddClassHandler((brush, e) => + { + Color baseColor; + if (!brush.IsSet(BaseColorProperty)) + { + baseColor = brush.Color; + SetBaseColor(brush, baseColor); + } + else + { + baseColor = brush.GetValue(BaseColorProperty); + } + var hsl = HslColor.FromRgba(baseColor).Scale(e.NewValue as HslDiff); + brush.Color = hsl.ToRgba(); + }); + } + + public static Color GetBaseColor(AvaloniaObject obj) + { + return obj.GetValue(BaseColorProperty); + } + + public static void SetBaseColor(AvaloniaObject obj, Color value) + { + obj.SetValue(BaseColorProperty, value); + } + + public static HslDiff GetHue(AvaloniaObject obj) + { + return obj.GetValue(HueProperty); + } + + public static void SetHue(AvaloniaObject obj, HslDiff value) + { + obj.SetValue(HueProperty, value); + } } \ No newline at end of file diff --git a/AvaloniaStyles/Controls/AlternativeScrollViewer.cs b/AvaloniaStyles/Controls/AlternativeScrollViewer.cs deleted file mode 100644 index 80c164723..000000000 --- a/AvaloniaStyles/Controls/AlternativeScrollViewer.cs +++ /dev/null @@ -1,37 +0,0 @@ -using Avalonia; -using Avalonia.Controls; -using Avalonia.Controls.Presenters; -using Avalonia.Input; - -namespace AvaloniaStyles.Controls -{ - public class AlternativeScrollViewer : ScrollViewer - { - private bool shouldHandleWheelEvent; - public static readonly DirectProperty ShouldHandleWheelEventProperty = AvaloniaProperty.RegisterDirect("ShouldHandleWheelEvent", o => o.ShouldHandleWheelEvent, (o, v) => o.ShouldHandleWheelEvent = v); - - public bool ShouldHandleWheelEvent - { - get => shouldHandleWheelEvent; - set => SetAndRaise(ShouldHandleWheelEventProperty, ref shouldHandleWheelEvent, value); - } - } - - public class AlternativeScrollContentPresenter : ScrollContentPresenter - { - private bool shouldHandleWheelEvent; - public static readonly DirectProperty ShouldHandleWheelEventProperty = AvaloniaProperty.RegisterDirect("ShouldHandleWheelEvent", o => o.ShouldHandleWheelEvent, (o, v) => o.ShouldHandleWheelEvent = v); - - public bool ShouldHandleWheelEvent - { - get => shouldHandleWheelEvent; - set => SetAndRaise(ShouldHandleWheelEventProperty, ref shouldHandleWheelEvent, value); - } - - protected override void OnPointerWheelChanged(PointerWheelEventArgs e) - { - base.OnPointerWheelChanged(e); - e.Handled = shouldHandleWheelEvent; - } - } -} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/BaseMessageBoxWindow.cs b/AvaloniaStyles/Controls/BaseMessageBoxWindow.cs index 948aae2a4..365472d50 100644 --- a/AvaloniaStyles/Controls/BaseMessageBoxWindow.cs +++ b/AvaloniaStyles/Controls/BaseMessageBoxWindow.cs @@ -10,8 +10,10 @@ namespace AvaloniaStyles.Controls { - public class BaseMessageBoxWindow : Window, IStyleable + public class BaseMessageBoxWindow : Window { + protected override Type StyleKeyOverride => typeof(BaseMessageBoxWindow); + public static readonly StyledProperty MessageProperty = AvaloniaProperty.Register(nameof(Message)); @@ -30,10 +32,10 @@ public string Header set => SetValue(HeaderProperty, value); } - public static readonly StyledProperty ImageProperty = - AvaloniaProperty.Register(nameof(Image)); + public static readonly StyledProperty ImageProperty = + AvaloniaProperty.Register(nameof(Image)); - public IControl Image + public Control Image { get => GetValue(ImageProperty); set => SetValue(ImageProperty, value); @@ -41,7 +43,6 @@ public IControl Image public BaseMessageBoxWindow() { - this.AttachDevTools(); if (RuntimeInformation.IsOSPlatform(OSPlatform.OSX)) PseudoClasses.Add(":macos"); else if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) @@ -49,17 +50,17 @@ public BaseMessageBoxWindow() else if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) PseudoClasses.Add(":linux"); - Win32.SetDarkMode(PlatformImpl.Handle.Handle, SystemTheme.EffectiveThemeIsDark); + if (TryGetPlatformHandle() is { } handle) + Win32.SetDarkMode(handle.Handle, SystemTheme.EffectiveThemeIsDark); } - - Type IStyleable.StyleKey => typeof(BaseMessageBoxWindow); protected override void OnApplyTemplate(TemplateAppliedEventArgs e) { base.OnApplyTemplate(e); ExtendClientAreaChromeHints = ExtendClientAreaChromeHints.NoChrome; if (Background is ISolidColorBrush brush) - Win32.SetTitleBarColor(PlatformImpl.Handle.Handle, brush.Color); + if (TryGetPlatformHandle() is { } handle) + Win32.SetTitleBarColor(handle.Handle, brush.Color); } } } \ No newline at end of file diff --git a/AvaloniaStyles/Controls/CompletionComboBox.axaml b/AvaloniaStyles/Controls/CompletionComboBox.axaml index 37a339bcf..788c98b89 100644 --- a/AvaloniaStyles/Controls/CompletionComboBox.axaml +++ b/AvaloniaStyles/Controls/CompletionComboBox.axaml @@ -45,8 +45,8 @@ MaxWidth="700" MaxHeight="600" PlacementConstraintAdjustment="SlideY,SlideX" - PlacementMode="Bottom" - PlacementGravity="Bottom" + Placement="BottomEdgeAlignedLeft" + PlacementGravity="BottomLeft" PlacementTarget="PART_Button"> - \ No newline at end of file + diff --git a/AvaloniaStyles/Controls/CompletionComboBox.axaml.cs b/AvaloniaStyles/Controls/CompletionComboBox.axaml.cs index b31b51bfe..161a49d7c 100644 --- a/AvaloniaStyles/Controls/CompletionComboBox.axaml.cs +++ b/AvaloniaStyles/Controls/CompletionComboBox.axaml.cs @@ -230,19 +230,23 @@ private void Copy() if (SelectedItem != null) { if (watermark is not null) - AvaloniaLocator.Current.GetRequiredService().SetTextAsync(watermark); + TopLevel.GetTopLevel(this)?.Clipboard?.SetTextAsync(watermark); } } private async Task Paste() { - var text = await AvaloniaLocator.Current.GetRequiredService().GetTextAsync(); + var textTask = TopLevel.GetTopLevel(this)?.Clipboard?.GetTextAsync(); + if (textTask == null) + return; + var text = await textTask; + //IsDropDownOpen = true; //await Task.Delay(1); - SearchText = text; + SearchText = text ?? ""; SearchTextBox.RaiseEvent(new KeyEventArgs() { - Device = null, + //Device = null, Handled = false, Key = Key.Enter, Route = RoutingStrategies.Tunnel, @@ -260,15 +264,13 @@ protected override void OnTextInput(TextInputEventArgs e) IsDropDownOpen = true; DispatcherTimer.RunOnce(() => { - SearchTextBox.RaiseEvent(new TextInputEventArgs - { - Device = e.Device, - Handled = false, - Text = e.Text, - Route = e.Route, - RoutedEvent = e.RoutedEvent, - Source = SearchTextBox - }); + TextInputEventArgs args = new TextInputEventArgs(); + args.Handled = false; + args.Text = e.Text; + args.Route = e.Route; + args.RoutedEvent = e.RoutedEvent; + args.Source = SearchTextBox; + SearchTextBox.RaiseEvent(args); }, TimeSpan.FromMilliseconds(1)); e.Handled = true; } @@ -302,7 +304,7 @@ static CompletionComboBox() return; //box.SearchText = ""; - FocusManager.Instance!.Focus(box.SearchTextBox, NavigationMethod.Pointer); + box.SearchTextBox.Focus(NavigationMethod.Pointer); box.SearchTextBox.SelectionEnd = box.SearchTextBox.SelectionStart = box.SearchText.Length; }, TimeSpan.FromMilliseconds(16)); @@ -318,12 +320,12 @@ static CompletionComboBox() protected override void OnApplyTemplate(TemplateAppliedEventArgs e) { - SearchTextBox = e.NameScope.Find("PART_SearchTextBox"); + SearchTextBox = e.NameScope.Find("PART_SearchTextBox") ?? throw new NullReferenceException("Couldn't find PART_SearchTextBox"); SearchTextBox.AddHandler(KeyDownEvent, SearchTextBoxOnKeyDown, RoutingStrategies.Tunnel); - ToggleButton = e.NameScope.Find("PART_Button"); + ToggleButton = e.NameScope.Find("PART_Button") ?? throw new NullReferenceException("Couldn't find PART_Button"); - ListBox = e.NameScope.Find("PART_SelectingItemsControl"); + ListBox = e.NameScope.Find("PART_SelectingItemsControl") ?? throw new NullReferenceException("Couldn't find PART_SelectingItemsControl"); ListBox.PointerReleased += OnSelectorPointerReleased; if (ListBox != null) { @@ -350,11 +352,11 @@ private void SearchTextBoxOnKeyDown(object? sender, KeyEventArgs e) return; if (watermark == null) return; - foreach (var binding in AvaloniaLocator.Current.GetRequiredService().Copy) + foreach (var binding in TopLevel.GetTopLevel(this)!.PlatformSettings!.HotkeyConfiguration!.Copy) { if (binding.Matches(e)) { - AvaloniaLocator.Current.GetRequiredService().SetTextAsync(watermark); + TopLevel.GetTopLevel(this)?.Clipboard?.SetTextAsync(watermark); e.Handled = true; } } @@ -362,7 +364,7 @@ private void SearchTextBoxOnKeyDown(object? sender, KeyEventArgs e) private void OnSelectorPointerReleased(object? sender, PointerReleasedEventArgs e) { - if (e.Source is IControl control && control.FindAncestorOfType() != null) + if (e.Source is Control control && control.FindAncestorOfType() != null) return; Commit(adapter?.SelectedItem); @@ -374,7 +376,7 @@ private void OnTextBoxTextChanged() Dispatcher.UIThread.Post(() => { // Call the central updated text method as a user-initiated action - TextUpdated(textBox.Text); + TextUpdated(textBox.Text ?? ""); }); } @@ -396,14 +398,14 @@ protected ISelectionAdapter SelectionAdapter { if (adapter != null) { - adapter.Items = null; + adapter.ItemsSource = null; } adapter = value; if (adapter != null) { - adapter.Items = view; + adapter.ItemsSource = view; } } } @@ -427,6 +429,23 @@ private TextBox SearchTextBox textBox.GetObservable(TextBox.TextProperty) .Skip(1) .Subscribe(_ => OnTextBoxTextChanged()) + .Combine( + textBox.AddDisposableHandler(TextInputEvent, (_, args) => + { + if (args.Text == " ") + { + if (SelectionAdapter.SelectedItem != null) + { + var container = + ListBox.ContainerFromIndex(ListBox.SelectedIndex); + if (container?.FindDescendantOfType() is CheckBox cb) + { + cb.IsChecked = !cb.IsChecked; + args.Handled = true; + } + } + } + }, RoutingStrategies.Tunnel)) .Combine( textBox.AddDisposableHandler(KeyDownEvent, (_, args) => { @@ -435,7 +454,7 @@ private TextBox SearchTextBox if (args.Key == Key.Enter) { - var enterArgs = new EnterPressedArgs(textBox.Text, + var enterArgs = new EnterPressedArgs(textBox.Text ?? "", SelectionAdapter.SelectedItem); OnEnterPressed?.Invoke(this, enterArgs); if (!enterArgs.Handled) @@ -445,8 +464,15 @@ private TextBox SearchTextBox } else if (args.Key == Key.Tab) { - args.Key = (args.KeyModifiers & KeyModifiers.Shift) != 0 ? Key.Up : Key.Down; - SelectionAdapter.HandleKeyDown(args); + var newArgs = new KeyEventArgs() + { + Key = (args.KeyModifiers & KeyModifiers.Shift) != 0 ? Key.Up : Key.Down, + Route = args.Route, + RoutedEvent = args.RoutedEvent, + KeyModifiers = args.KeyModifiers, + Source = args.Source, + }; + SelectionAdapter.HandleKeyDown(newArgs); args.Handled = true; } else if (args.Key == Key.Escape) @@ -454,20 +480,6 @@ private TextBox SearchTextBox Close(); args.Handled = true; } - else if (args.Key == Key.Space) - { - if (SelectionAdapter.SelectedItem != null) - { - var container = - ListBox.ItemContainerGenerator - .ContainerFromIndex(ListBox.SelectedIndex); - if (container?.FindDescendantOfType() is CheckBox cb) - { - cb.IsChecked = !cb.IsChecked; - args.Handled = true; - } - } - } else { SelectionAdapter.HandleKeyDown(args); @@ -478,7 +490,7 @@ private TextBox SearchTextBox // don't ever let textbox lost focus if (IsDropDownOpen) { - Dispatcher.UIThread.Post(() => FocusManager.Instance!.Focus(textBox)); + Dispatcher.UIThread.Post(() => textBox.Focus()); } })); } @@ -498,7 +510,7 @@ private void Commit(object? item) private void Close() { IsDropDownOpen = false; - FocusManager.Instance!.Focus(ToggleButton, NavigationMethod.Tab); + ToggleButton.Focus(NavigationMethod.Tab); Closed?.Invoke(); } @@ -542,7 +554,12 @@ private async Task PopulateAsync(string searchText, CancellationToken cancellati { try { - IEnumerable result = await asyncPopulator.Invoke(items ?? Array.Empty(), searchText, cancellationToken); + IEnumerable? result = await asyncPopulator.Invoke(items ?? Array.Empty(), searchText, cancellationToken); + if (result == null) + { + view.Clear(); + return; + } var resultList = result is IList ? (IList)result : result.Cast()?.ToList(); if (cancellationToken.IsCancellationRequested) @@ -567,4 +584,4 @@ await Dispatcher.UIThread.InvokeAsync(() => } } } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/ControlRenderer.cs b/AvaloniaStyles/Controls/ControlRenderer.cs new file mode 100644 index 000000000..e7c084d31 --- /dev/null +++ b/AvaloniaStyles/Controls/ControlRenderer.cs @@ -0,0 +1,89 @@ +using System; +using System.Collections.Generic; +using System.Collections.Specialized; +using System.Linq; +using Avalonia; +using Avalonia.Controls; +using Avalonia.Media; +using Avalonia.Threading; + +namespace AvaloniaStyles.Controls; + +public class ControlRenderer : Control +{ + public Action? RenderOverride { get; set; } + + public override void Render(DrawingContext context) + { + RenderOverride?.Invoke(context); + } +} + +public abstract class RenderedPanel : Panel +{ + private ControlRenderer renderer; + + public Control Renderer => renderer; + + public RenderedPanel() + { + renderer = new ControlRenderer(); + Children.Add(renderer); + renderer.RenderOverride = Render; + + Children.CollectionChanged += (sender, args) => + { + if (args.Action == NotifyCollectionChangedAction.Reset) + { + Children.Add(renderer); + } + else if (args.Action == NotifyCollectionChangedAction.Remove && + ReferenceEquals(args.OldItems![0], renderer)) + { + Children.Add(renderer); + } + }; + } + + protected void EnsureRenderer() + { + if (!Children.Contains(renderer)) + Children.Add(renderer); + } + + public IEnumerable ActualChildren => Children.Where(x => !IsRenderer(x)); + + public IEnumerable ActualVisualChildren => VisualChildren.Where(x => !IsRenderer(x)); + + // might be not the best solution? + protected override void ArrangeCore(Rect finalRect) + { + base.ArrangeCore(finalRect); + renderer.Arrange(new Rect(0, 0, finalRect.Width, finalRect.Height)); + } + + protected bool IsRenderer(Control control) + { + return ReferenceEquals(control, renderer); + } + + protected bool IsRenderer(Visual visual) + { + return ReferenceEquals(visual, renderer); + } + + protected static void AffectsRender(AvaloniaProperty property) where T : RenderedPanel + { + property.Changed.AddClassHandler((t, e) => + { + t.InvalidateVisual(); + }); + } + + public new void InvalidateVisual() + { + renderer.InvalidateVisual(); + } + + public new abstract void Render(DrawingContext context); +} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/DismissPopupBehaviour.cs b/AvaloniaStyles/Controls/DismissPopupBehaviour.cs index f3e7fb57f..ca23dda6f 100644 --- a/AvaloniaStyles/Controls/DismissPopupBehaviour.cs +++ b/AvaloniaStyles/Controls/DismissPopupBehaviour.cs @@ -22,7 +22,7 @@ protected override void OnDetaching() private void Handler(object? sender, PointerReleasedEventArgs e) { var control = (sender as Control); - var popup = control.FindLogicalAncestorOfType(); + var popup = control?.FindLogicalAncestorOfType(); if (popup != null) popup.IsOpen = false; } diff --git a/AvaloniaStyles/Controls/DropDownButton.axaml b/AvaloniaStyles/Controls/DropDownButton.axaml deleted file mode 100644 index dea1106fd..000000000 --- a/AvaloniaStyles/Controls/DropDownButton.axaml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - diff --git a/AvaloniaStyles/Controls/DropDownButton.axaml.cs b/AvaloniaStyles/Controls/DropDownButton.axaml.cs deleted file mode 100644 index a74256f4c..000000000 --- a/AvaloniaStyles/Controls/DropDownButton.axaml.cs +++ /dev/null @@ -1,37 +0,0 @@ -using Avalonia; -using Avalonia.Controls; -using Avalonia.Controls.Primitives; -using Avalonia.Metadata; - -namespace AvaloniaStyles.Controls -{ - public class DropDownButton : TemplatedControl - { - private object? button; - public static readonly DirectProperty ButtonProperty = AvaloniaProperty.RegisterDirect("Button", o => o.Button, (o, v) => o.Button = v); - private bool isDropDownOpened; - public static readonly DirectProperty IsDropDownOpenedProperty = AvaloniaProperty.RegisterDirect("IsDropDownOpened", o => o.IsDropDownOpened, (o, v) => o.IsDropDownOpened = v); - - public static readonly StyledProperty ChildProperty = - AvaloniaProperty.Register(nameof(Child)); - - public object? Button - { - get => button; - set => SetAndRaise(ButtonProperty, ref button, value); - } - - public bool IsDropDownOpened - { - get => isDropDownOpened; - set => SetAndRaise(IsDropDownOpenedProperty, ref isDropDownOpened, value); - } - - [Content] - public Control? Child - { - get => GetValue(ChildProperty); - set => SetValue(ChildProperty, value); - } - } -} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/ExtendedWindow.cs b/AvaloniaStyles/Controls/ExtendedWindow.cs index 11c41dec3..f9ba12a6f 100644 --- a/AvaloniaStyles/Controls/ExtendedWindow.cs +++ b/AvaloniaStyles/Controls/ExtendedWindow.cs @@ -10,12 +10,11 @@ using Avalonia.Platform; using Avalonia.Styling; using Avalonia.Threading; -using Avalonia.Win32; using AvaloniaStyles.Utils; namespace AvaloniaStyles.Controls { - public class ExtendedWindow : Window, IStyleable + public class ExtendedWindow : Window { public static readonly StyledProperty ManagedIconProperty = AvaloniaProperty.Register(nameof(ManagedIcon)); @@ -26,8 +25,8 @@ public class ExtendedWindow : Window, IStyleable public static readonly StyledProperty ToolBarProperty = AvaloniaProperty.Register(nameof(ToolBar)); - public static readonly StyledProperty SideBarProperty = - AvaloniaProperty.Register(nameof(SideBar)); + public static readonly StyledProperty SideBarProperty = + AvaloniaProperty.Register(nameof(SideBar)); public static readonly StyledProperty StatusBarProperty = AvaloniaProperty.Register(nameof(StatusBar)); @@ -35,8 +34,8 @@ public class ExtendedWindow : Window, IStyleable public static readonly StyledProperty TabStripProperty = AvaloniaProperty.Register(nameof(TabStrip)); - public static readonly StyledProperty OverlayProperty = - AvaloniaProperty.Register(nameof(Overlay)); + public static readonly StyledProperty OverlayProperty = + AvaloniaProperty.Register(nameof(Overlay)); public static readonly StyledProperty SubTitleProperty = AvaloniaProperty.Register(nameof(SubTitle)); @@ -59,7 +58,7 @@ public ToolBar ToolBar set => SetValue(ToolBarProperty, value); } - public IControl? SideBar + public Control? SideBar { get => GetValue(SideBarProperty); set => SetValue(SideBarProperty, value); @@ -71,7 +70,7 @@ public StatusBar StatusBar set => SetValue(StatusBarProperty, value); } - public IControl Overlay + public Control Overlay { get => GetValue(OverlayProperty); set => SetValue(OverlayProperty, value); @@ -89,7 +88,7 @@ public TabStrip TabStrip set => SetValue(TabStripProperty, value); } - Type IStyleable.StyleKey => typeof(ExtendedWindow); + protected override Type StyleKeyOverride => typeof(ExtendedWindow); static ExtendedWindow() { @@ -121,38 +120,34 @@ static ExtendedWindow() BackgroundProperty.Changed.AddClassHandler((window, e) => { - window.BindBackgroundBrush(e.NewValue as Brush); + window.BindBackgroundBrush(e.NewValue as SolidColorBrush); }); } - private Brush? _backgroundBrush; + private IDisposable? backgroundBrushBinding; private void UnbindBackgroundBrush() { - if (_backgroundBrush != null) - { - _backgroundBrush.Invalidated -= BackgroundBrushOnInvalidated; - _backgroundBrush = null; - } + backgroundBrushBinding?.Dispose(); + backgroundBrushBinding = null; } - private void BindBackgroundBrush(Brush? brush) + private void BindBackgroundBrush(SolidColorBrush? brush) { UnbindBackgroundBrush(); - - _backgroundBrush = brush; - + if (brush != null) { - brush.Invalidated += BackgroundBrushOnInvalidated; - BackgroundBrushOnInvalidated(null, EventArgs.Empty); + backgroundBrushBinding = brush.GetObservable(SolidColorBrush.ColorProperty) + .Subscribe(_ => BackgroundBrushOnInvalidated(null, EventArgs.Empty)); } } private void BackgroundBrushOnInvalidated(object? sender, EventArgs e) { - if (Background is ISolidColorBrush brush && PlatformImpl != null) - Win32.SetTitleBarColor(PlatformImpl.Handle.Handle, brush.Color); + if (Background is ISolidColorBrush brush) + if (TryGetPlatformHandle() is { } handle) + Win32.SetTitleBarColor(handle.Handle, brush.Color); } private void UpdateChromeHints(ExtendedWindowChrome chrome) @@ -178,7 +173,8 @@ public ExtendedWindow() PseudoClasses.Add(":macos"); else if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) { - Win32.SetDarkMode(PlatformImpl.Handle.Handle, SystemTheme.EffectiveThemeIsDark); + if (TryGetPlatformHandle() is { } handle) + Win32.SetDarkMode(handle.Handle, SystemTheme.EffectiveThemeIsDark); PseudoClasses.Add(":windows"); if (Environment.OSVersion.Version.Build >= 22000) PseudoClasses.Add(":win11"); @@ -194,12 +190,14 @@ private static void UpdateScaling(Window window) if (!SystemTheme.CustomScalingValue.HasValue) { var primaryScreen = window.Screens.Primary ?? window.Screens.All.FirstOrDefault(); - scaling = (primaryScreen?.PixelDensity ?? 1); + scaling = (primaryScreen?.Scaling ?? 1); } else scaling = SystemTheme.CustomScalingValue.Value; var impl = window.PlatformImpl; + if (impl == null) + return; var f = impl.GetType().GetField("_scaling", BindingFlags.Instance | BindingFlags.NonPublic); if (f != null) { @@ -209,7 +207,8 @@ private static void UpdateScaling(Window window) var oldWidth = window.Width * curVal; var oldHeight = window.Height * curVal; f.SetValue(impl, scaling); - impl.ScalingChanged(scaling); + Action? scalingChanged = (Action?)impl.GetType().GetProperty("ScalingChanged", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)?.GetValue(impl); + scalingChanged?.Invoke(scaling); DispatcherTimer.RunOnce(() => { window.Width = oldWidth / scaling; @@ -288,4 +287,4 @@ public enum ExtendedWindowChrome AlwaysSystemChrome, MacOsChrome } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/Extensions.cs b/AvaloniaStyles/Controls/Extensions.cs index 5048c0310..177188f1a 100644 --- a/AvaloniaStyles/Controls/Extensions.cs +++ b/AvaloniaStyles/Controls/Extensions.cs @@ -13,7 +13,7 @@ using Avalonia.Layout; using AvaloniaStyles.Converters; using FuzzySharp; -using JetBrains.Annotations; +using WDE.MVVM; using WDE.MVVM.Observable; namespace AvaloniaStyles.Controls @@ -22,13 +22,13 @@ internal class EnumToIntConverter : IValueConverter { public static EnumToIntConverter Instance { get; } = new(); - public object Convert(object? value, Type targetType, object? parameter, CultureInfo culture) + public object? Convert(object? value, Type targetType, object? parameter, CultureInfo culture) { var val = System.Convert.ToUInt32(value); return $"{value} ({val})"; } - public object ConvertBack(object? value, Type targetType, object? parameter, CultureInfo culture) + public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo culture) { throw new NotImplementedException(); } @@ -56,11 +56,10 @@ private static bool IsFlagEnum(Type type) return type.GetCustomAttribute() != null; } - internal sealed class Option : INotifyPropertyChanged + internal sealed class Option : ObservableBase, INotifyPropertyChanged { private readonly Type type; private readonly WeakReference completionBoxReference; - private IDisposable? disposable; public Option(object enumValue, Type type, CompletionComboBox combo, string text) { @@ -71,7 +70,6 @@ public Option(object enumValue, Type type, CompletionComboBox combo, string text EnumInteger = Convert.ToUInt32(enumValue); TextWithNumber = $"{Text} {EnumInteger}"; - subscribed = 0; } public string TextWithNumber { get; } @@ -112,48 +110,15 @@ public bool IsChecked } } - private event PropertyChangedEventHandler? PropertyChanged; - - private int subscribed; - event PropertyChangedEventHandler? INotifyPropertyChanged.PropertyChanged { - add - { - if (!completionBoxReference.TryGetTarget(out var completionBox)) - return; - - subscribed++; - PropertyChanged += value; - if (subscribed > 0) - { - if (disposable != null) - throw new Exception(); - disposable = completionBox.GetObservable(CompletionComboBox.SelectedItemProperty).SubscribeAction(_ => - { - OnPropertyChanged(nameof(IsChecked)); - }); - } - } - remove - { - if (!completionBoxReference.TryGetTarget(out _)) - return; - - PropertyChanged -= value; - subscribed--; - if (subscribed == 0) - { - disposable?.Dispose(); - disposable = null; - } - } + add => PropertyChanged += value; + remove => PropertyChanged -= value; } - [NotifyPropertyChangedInvocator] - private void OnPropertyChanged([CallerMemberName] string? propertyName = null) + public void RaiseIsChecked(CompletionComboBox combo) { - PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName)); + RaisePropertyChanged(nameof(IsChecked)); } } @@ -165,6 +130,11 @@ static Extensions() return; var values = Enum.GetValuesAsUnderlyingType(type).Cast().Zip(Enum.GetNames(type), (val, name) => new Option(val, type, combo, name)).ToList(); + combo.GetObservable(CompletionComboBox.SelectedItemProperty).SubscribeAction(_ => + { + foreach (var val in values) + val.RaiseIsChecked(combo); + }); combo.AsyncPopulator = async (items, str, _) => { if (string.IsNullOrEmpty(str)) @@ -232,4 +202,4 @@ static Extensions() }); } } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/FastTableView/ColorsDark.axaml b/AvaloniaStyles/Controls/FastTableView/ColorsDark.axaml index 5930a1d54..219e2d786 100644 --- a/AvaloniaStyles/Controls/FastTableView/ColorsDark.axaml +++ b/AvaloniaStyles/Controls/FastTableView/ColorsDark.axaml @@ -2,14 +2,14 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> - + - + diff --git a/AvaloniaStyles/Controls/FastTableView/ColorsLight.axaml b/AvaloniaStyles/Controls/FastTableView/ColorsLight.axaml index e46ffca0e..df33bd30c 100644 --- a/AvaloniaStyles/Controls/FastTableView/ColorsLight.axaml +++ b/AvaloniaStyles/Controls/FastTableView/ColorsLight.axaml @@ -2,14 +2,14 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> - + - + diff --git a/AvaloniaStyles/Controls/FastTableView/Extensions.cs b/AvaloniaStyles/Controls/FastTableView/Extensions.cs index 128712cc8..57e188e57 100644 --- a/AvaloniaStyles/Controls/FastTableView/Extensions.cs +++ b/AvaloniaStyles/Controls/FastTableView/Extensions.cs @@ -1,4 +1,5 @@ using Avalonia; +using Avalonia.Styling; namespace AvaloniaStyles.Controls.FastTableView; @@ -7,12 +8,12 @@ internal static partial class Extensions public static bool GetResource(this object? _, string key, T defaultVal, out T outT) { outT = defaultVal; - if ((Application.Current?.Styles.TryGetResource(key, out var res) ?? false) && res is T t) + if ((Application.Current?.Styles.TryGetResource(key, SystemTheme.EffectiveThemeVariant, out var res) ?? false) && res is T t) { outT = t; return true; } - if ((Application.Current?.Resources.TryGetResource(key, out var res2) ?? false) && res2 is T t2) + if ((Application.Current?.Resources.TryGetResource(key, SystemTheme.EffectiveThemeVariant, out var res2) ?? false) && res2 is T t2) { outT = t2; return true; diff --git a/AvaloniaStyles/Controls/FastTableView/ICustomCellDrawer.cs b/AvaloniaStyles/Controls/FastTableView/ICustomCellDrawer.cs index f3cbac767..ae903abe6 100644 --- a/AvaloniaStyles/Controls/FastTableView/ICustomCellDrawer.cs +++ b/AvaloniaStyles/Controls/FastTableView/ICustomCellDrawer.cs @@ -1,3 +1,4 @@ +using System.Globalization; using System.Threading.Tasks; using Avalonia; using Avalonia.Media; @@ -190,15 +191,13 @@ public bool UpdateCursor(Point point, bool leftPressed) protected void DrawText(DrawingContext context, Rect rect, string text, Color? color = null, float opacity = 0.4f) { - var ft = new FormattedText + var textColor = new SolidColorBrush(color ?? TextBrush.Color, opacity); + var ft = new FormattedText(text, CultureInfo.InvariantCulture, FlowDirection.LeftToRight, new Typeface(MonoFont), 12, textColor) { - Text = text, - Constraint = new Size(rect.Width, rect.Height), - Typeface = new Typeface(MonoFont),//Typeface.Default, - FontSize = 12 + MaxTextHeight = rect.Height, + MaxTextWidth = float.MaxValue // rect.Width // we don't want text wrapping so pass float.MaxValue }; - var textColor = new SolidColorBrush(color ?? TextBrush.Color, opacity); - context.DrawText(textColor, new Point(rect.X, rect.Center.Y - ft.Bounds.Height / 2), ft); + context.DrawText(ft, new Point(rect.X, rect.Center.Y - ft.Height / 2)); } private const double CheckBoxSize = 20; @@ -261,28 +260,18 @@ protected void DrawButton(DrawingContext context, Rect rect, string text, int ma var state = context.PushClip(rect); if (char.IsAscii(text[0])) { - var ft = new FormattedText + var ft = new FormattedText(text, CultureInfo.InvariantCulture, FlowDirection.LeftToRight, Typeface.Default, 12, ButtonTextPen.Brush) { - Text = text, - Constraint = new Size(rect.Width, rect.Height), - Typeface = Typeface.Default, - FontSize = 12 + MaxTextHeight = rect.Height, + MaxTextWidth = float.MaxValue // rect.Width // we don't want text wrapping so pass float.MaxValue }; - context.DrawText(ButtonTextPen.Brush, new Point(rect.Center.X - ft.Bounds.Width / 2, rect.Center.Y - ft.Bounds.Height / 2), ft); + context.DrawText(ft, new Point(rect.Center.X - ft.Width / 2, rect.Center.Y - ft.Height / 2)); } else { var tl = new TextLayout(text, Typeface.Default, 12, ButtonTextPen.Brush, TextAlignment.Center, maxWidth: rect.Width, maxHeight:rect.Height); - Draw(tl, context, new Point(rect.Left, rect.Center.Y - tl.Size.Height / 2)); + tl.Draw(context, new Point(rect.Left, rect.Center.Y - tl.Height / 2)); } state.Dispose(); } - - /* not required in avalonia 11, just call layout.Draw(context, p) */ - private static void Draw(TextLayout layout, DrawingContext context, Point p) - { - double offset = layout.MaxWidth / 2 - layout.Size.Width / 2; - using var _ = context.PushPostTransform(Matrix.CreateTranslation(p.X + offset, p.Y)); - layout.Draw(context); - } } \ No newline at end of file diff --git a/AvaloniaStyles/Controls/FastTableView/PhantomTextBox.cs b/AvaloniaStyles/Controls/FastTableView/PhantomTextBox.cs index 94c1186f5..cb711c8f5 100644 --- a/AvaloniaStyles/Controls/FastTableView/PhantomTextBox.cs +++ b/AvaloniaStyles/Controls/FastTableView/PhantomTextBox.cs @@ -19,7 +19,6 @@ namespace AvaloniaStyles.Controls.FastTableView; public abstract class PhantomControlBase where T : Control { private AdornerLayer? adornerLayer = null; - private Panel panel = null!; private Visual? parent = null!; private IDisposable? clickDisposable = null; private IDisposable? focusDisposable = null; @@ -102,7 +101,7 @@ protected void Despawn(bool save) focusDisposable = null; if (parent is IInputElement inputElement) - FocusManager.Instance!.Focus(inputElement); + inputElement.Focus(); element = null; parent = null; IsOpened = false; @@ -122,7 +121,7 @@ public enum ActionAfterSave private Action? currentOnApply = null; private ActionAfterSave actionAfterSave; - public void Spawn(Visual parent, Rect position, string text, bool selectAll, Action onApply, FontFamily? customFont = null) + public void Spawn(Visual parent, Rect position, string text, bool selectAll, Action onApply) { Despawn(false); @@ -135,7 +134,7 @@ public void Spawn(Visual parent, Rect position, string text, bool selectAll, Act textBox.Padding = new Thickness(5 + 3, -2, 0, 0); textBox.Margin = new Thickness(0, 0, 0, 0); textBox.Text = text; - textBox.FontFamily = customFont ?? new FontFamily("Consolas,Menlo,Courier,Courier New"); + textBox.FontFamily = new FontFamily("Consolas,Menlo,Courier,Courier New"); textBox.KeyBindings.Add(new KeyBinding() { Gesture = new KeyGesture(Key.Enter), @@ -181,8 +180,8 @@ public void Spawn(Visual parent, Rect position, string text, bool selectAll, Act return; textBox.LostFocus += ElementLostFocus; - - DispatcherTimer.RunOnce(textBox.Focus, TimeSpan.FromMilliseconds(1)); + + DispatcherTimer.RunOnce(() => textBox.Focus(), TimeSpan.FromMilliseconds(1)); if (selectAll) textBox.SelectAll(); else @@ -203,7 +202,7 @@ protected override void Cleanup(TextBox element) protected override void Save(TextBox element) { - currentOnApply?.Invoke(element.Text, actionAfterSave); + currentOnApply?.Invoke(element.Text ?? "", actionAfterSave); } } @@ -240,7 +239,7 @@ protected void Spawn(Visual parent, Rect position, string? searchText, IEnumerab flagsComboBox.IsLightDismissEnabled = false; // we are handling it ourselves, without doing .Handled = true so that as soon as user press outside of popup, the click is treated as actual click flagsComboBox.Closed += CompletionComboBoxOnClosed; - if (!AttachAsAdorner(parent, position, flagsComboBox)) + if (!AttachAsAdorner(parent, position, flagsComboBox, true)) return; DispatcherTimer.RunOnce(() => @@ -259,4 +258,4 @@ protected override void Cleanup(CompletionComboBox element) { element.Closed -= CompletionComboBoxOnClosed; } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/FastTableView/Row.cs b/AvaloniaStyles/Controls/FastTableView/Row.cs index dab9cc28d..3062ef0dc 100644 --- a/AvaloniaStyles/Controls/FastTableView/Row.cs +++ b/AvaloniaStyles/Controls/FastTableView/Row.cs @@ -1,11 +1,9 @@ using System; using System.Collections.Generic; -using System.Collections.ObjectModel; using System.ComponentModel; using System.Globalization; using System.Linq; using System.Runtime.CompilerServices; -using JetBrains.Annotations; namespace AvaloniaStyles.Controls.FastTableView; @@ -86,7 +84,6 @@ public void UpdateFromString(string newValue) public string? StringValue => value.ToString(); - [NotifyPropertyChangedInvocator] protected virtual void OnPropertyChanged([CallerMemberName] string? propertyName = null) { PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName)); diff --git a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Arrange.cs b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Arrange.cs index d233f93ea..164214834 100644 --- a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Arrange.cs +++ b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Arrange.cs @@ -28,7 +28,7 @@ protected override Size ArrangeOverride(Size finalSize) } headerViews.Reset(template); - subheaderViews.Reset(AdditionalGroupSubHeaderTemplate); + subheaderViews.Reset(AdditionalGroupSubHeader); if (!IsGroupingEnabled) { diff --git a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Data.cs b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Data.cs index 64149458c..ccbcffe61 100644 --- a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Data.cs +++ b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Data.cs @@ -2,6 +2,7 @@ using System.Collections.Generic; using System.Collections.ObjectModel; using System.Collections.Specialized; +using Avalonia.Threading; using DynamicData; using WDE.Common.Utils; @@ -61,18 +62,18 @@ private void FixSelectedRowIndexIfInvalid() return; if (Items == null) { - SelectedRowIndex = VerticalCursor.None; + SetCurrentValue(SelectedRowIndexProperty, VerticalCursor.None); return; } if (SelectedRowIndex.GroupIndex >= Items.Count) { - SelectedRowIndex = VerticalCursor.None; + SetCurrentValue(SelectedRowIndexProperty, VerticalCursor.None); return; } if (SelectedRowIndex.RowIndex >= Items[SelectedRowIndex.GroupIndex].Rows.Count) - SelectedRowIndex = VerticalCursor.None; + SetCurrentValue(SelectedRowIndexProperty, VerticalCursor.None); } private void ItemsChanged(object? sender, NotifyCollectionChangedEventArgs e) @@ -111,7 +112,9 @@ private void RowChanged(ITableRowGroup group, ITableRow obj) var rowIndex = Items[groupIndex].Rows.IndexOf(obj); if (IsRowVisible(new VerticalCursor(groupIndex, rowIndex))) - InvalidateVisual(); + { + Dispatcher.UIThread.Post(InvalidateVisual); + } } diff --git a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Navigation.cs b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Navigation.cs index 5314487b4..a7326ccf6 100644 --- a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Navigation.cs +++ b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Navigation.cs @@ -39,10 +39,10 @@ private bool MoveCursorRight() if (TableSpan is { } span && span.IsMerged(SelectedRowIndex.GroupIndex, SelectedRowIndex.RowIndex, next.Value, out var firstRow, out var firstColumn)) { next = firstColumn; - SelectedRowIndex = new VerticalCursor(SelectedRowIndex.GroupIndex, firstRow); + SetCurrentValue(SelectedRowIndexProperty, new VerticalCursor(SelectedRowIndex.GroupIndex, firstRow)); } - SelectedCellIndex = next.Value; + SetCurrentValue(SelectedCellIndexProperty, next.Value); return true; } @@ -55,10 +55,10 @@ private bool MoveCursorLeft() if (TableSpan is { } span && span.IsMerged(SelectedRowIndex.GroupIndex, SelectedRowIndex.RowIndex, prev.Value, out var firstRow, out var firstColumn)) { prev = firstColumn; - SelectedRowIndex = new VerticalCursor(SelectedRowIndex.GroupIndex, firstRow); + SetCurrentValue(SelectedRowIndexProperty, new VerticalCursor(SelectedRowIndex.GroupIndex, firstRow)); } - SelectedCellIndex = prev.Value; + SetCurrentValue(SelectedCellIndexProperty, prev.Value); return true; } @@ -118,7 +118,7 @@ int FixRowToFirstMerged(int group, int row) { if (IsFilteredRowVisible(items[group], items[group].Rows[row], rowFilter, rowFilterParameter)) { - SelectedRowIndex = new VerticalCursor(group, row); + SetCurrentValue(SelectedRowIndexProperty, new VerticalCursor(group, row)); return true; } } @@ -148,7 +148,7 @@ private bool MoveCursorPrevious() if (MoveCursorLeft()) return true; - SelectedCellIndex = ColumnsCount - 1; + SetCurrentValue(SelectedCellIndexProperty, ColumnsCount - 1); return MoveCursorUp(); } @@ -156,16 +156,10 @@ private bool MoveCursorNext() { if (MoveCursorRight()) return true; - SelectedCellIndex = 0; + SetCurrentValue(SelectedCellIndexProperty, 0); return MoveCursorDown(); } - - public void SetOwner(IInputRoot owner) - { - - } - public void Move(IInputElement element, NavigationDirection direction, KeyModifiers keyModifiers = KeyModifiers.None) { switch (direction) @@ -177,12 +171,12 @@ public void Move(IInputElement element, NavigationDirection direction, KeyModifi MoveCursorPrevious(); break; case NavigationDirection.First: - SelectedRowIndex = new VerticalCursor(0, 0); - SelectedCellIndex = 0; + SetCurrentValue(SelectedRowIndexProperty, new VerticalCursor(0, 0)); + SetCurrentValue(SelectedCellIndexProperty, 0); break; case NavigationDirection.Last: - SelectedRowIndex = new VerticalCursor(Items?.Count ?? 0 - 1, Items?[^1]?.Rows?.Count ?? 0 - 1); - SelectedCellIndex = ColumnsCount - 1; + SetCurrentValue(SelectedRowIndexProperty, new VerticalCursor(Items?.Count ?? 0 - 1, Items?[^1]?.Rows?.Count ?? 0 - 1)); + SetCurrentValue(SelectedCellIndexProperty, ColumnsCount - 1); break; case NavigationDirection.Left: MoveCursorLeft(); diff --git a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Properties.cs b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Properties.cs index 653832e2e..d911d69b2 100644 --- a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Properties.cs +++ b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Properties.cs @@ -30,20 +30,20 @@ public partial class VeryFastTableView public static readonly StyledProperty CustomCellInteractorProperty = AvaloniaProperty.Register(nameof(CustomCellInteractor)); public static readonly StyledProperty MultiSelectionProperty = AvaloniaProperty.Register(nameof(MultiSelection), defaultValue: new TableMultiSelection()); public static readonly StyledProperty GroupHeaderTemplateProperty = AvaloniaProperty.Register(nameof(GroupHeaderTemplate)); - public static readonly StyledProperty AdditionalGroupSubHeaderProperty = AvaloniaProperty.Register(nameof(AdditionalGroupSubHeaderTemplate)); + public static readonly StyledProperty AdditionalGroupSubHeaderProperty = AvaloniaProperty.Register(nameof(AdditionalGroupSubHeader)); public static readonly StyledProperty InteractiveHeaderProperty = AvaloniaProperty.Register(nameof(InteractiveHeader)); - public static readonly StyledProperty RowFilterParameterProperty = AvaloniaProperty.Register("RowFilterParameter"); - public static readonly StyledProperty IsGroupingEnabledProperty = AvaloniaProperty.Register("IsGroupingEnabled"); + public static readonly StyledProperty RowFilterParameterProperty = AvaloniaProperty.Register(nameof(RowFilterParameter)); + public static readonly StyledProperty IsGroupingEnabledProperty = AvaloniaProperty.Register(nameof(IsGroupingEnabled)); public static readonly StyledProperty SubHeaderHeightProperty = AvaloniaProperty.Register(nameof(SubHeaderHeight), defaultValue: 0); - public static readonly StyledProperty IsDynamicHeaderHeightProperty = AvaloniaProperty.Register("DynamicRowHeight"); + public static readonly StyledProperty IsDynamicHeaderHeightProperty = AvaloniaProperty.Register(nameof(IsDynamicHeaderHeight)); - public static readonly RoutedEvent ColumnPressedEvent = RoutedEvent.Register("ColumnPressed", RoutingStrategies.Tunnel | RoutingStrategies.Bubble); + public static readonly RoutedEvent ColumnPressedEvent = RoutedEvent.Register(nameof(ColumnPressed), RoutingStrategies.Tunnel | RoutingStrategies.Bubble); private List columnVisibility = new List(); - public static readonly StyledProperty RowFilterProperty = AvaloniaProperty.Register("RowFilter"); - public static readonly StyledProperty IsReadOnlyProperty = AvaloniaProperty.Register("IsReadOnly"); + public static readonly StyledProperty RowFilterProperty = AvaloniaProperty.Register(nameof(RowFilter)); + public static readonly StyledProperty IsReadOnlyProperty = AvaloniaProperty.Register(nameof(IsReadOnly)); public static readonly StyledProperty TableSpanProperty = AvaloniaProperty.Register(nameof(TableSpan)); @@ -120,7 +120,7 @@ public IDataTemplate? GroupHeaderTemplate set => SetValue(GroupHeaderTemplateProperty, value); } - public IDataTemplate? AdditionalGroupSubHeaderTemplate + public IDataTemplate? AdditionalGroupSubHeader { get => GetValue(AdditionalGroupSubHeaderProperty); set => SetValue(AdditionalGroupSubHeaderProperty, value); diff --git a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Rendering.cs b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Rendering.cs index bc982d7bf..6726a6312 100644 --- a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Rendering.cs +++ b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.Rendering.cs @@ -1,8 +1,11 @@ using System; +using System.Globalization; using System.Linq; using Avalonia; using Avalonia.Controls; +using Avalonia.Controls.Documents; using Avalonia.Media; +using Avalonia.Styling; using WDE.Common.Utils; using WDE.MVVM.Observable; @@ -19,7 +22,7 @@ private Rect DataViewport { var scrollViewer = ScrollViewer; if (scrollViewer == null) - return Rect.Empty; + return default; return new Rect(scrollViewer.Offset.X, scrollViewer.Offset.Y + DrawingStartOffsetY, scrollViewer.Viewport.Width, scrollViewer.Viewport.Height - DrawingStartOffsetY); } } @@ -45,11 +48,10 @@ protected bool IsFilteredGroupVisible(ITableRowGroup group, IRowFilterPredicate? public override void Render(DrawingContext context) { - base.Render(context); if (Items == null) return; - var font = new Typeface(TextBlock.GetFontFamily(this)); + var font = new Typeface(TextElement.GetFontFamily(this)); var actualWidth = Bounds.Width; @@ -104,13 +106,13 @@ public override void Render(DrawingContext context) } // out of bounds in the upper part - if (groupStartY + groupHeight < DataViewport.Top) + if (groupStartY + groupHeight < viewPort.Top) { if (group.Rows.Count % 2 == 1) odd = !odd; y += groupHeight; } - else if (groupStartY > DataViewport.Bottom) // below + else if (groupStartY > viewPort.Bottom) // below { break; } @@ -162,7 +164,7 @@ public override void Render(DrawingContext context) // we draw only the visible columns // todo: could be optimized so we don't iterate through all columns when we know we don't need to - if (x + columnWidth > DataViewport.Left && x < DataViewport.Right) + if (x + columnWidth > viewPort.Left && x < viewPort.Right) { var rect = span == null ? new Rect(x, y, columnWidth, RowHeight) : CellRect(cellIndex, new VerticalCursor(groupIndex, rowIndex)); var rectWidth = rect.Width; @@ -177,26 +179,21 @@ public override void Render(DrawingContext context) var indexOfEndOfLine = text.IndexOf('\n'); if (indexOfEndOfLine != -1) text = text.Substring(0, indexOfEndOfLine); - + rect = rect.WithWidth(rect.Width - ColumnSpacing); - var ft = new FormattedText - { - Text = text, - Constraint = new Size(rect.Width, rect.Height), - Typeface = font, - FontSize = 12 - }; + var ft = new FormattedText( + text, CultureInfo.CurrentCulture, FlowDirection.LeftToRight, font, 12, textColor); + ft.MaxTextWidth = float.MaxValue; // rect.Width; // we don't want to wrap the text, so pass large width + ft.MaxTextHeight = rect.Height; if (Math.Abs(rectWidth - rect.Width) > 0.01) { state.Dispose(); state = context.PushClip(rect); } - - context.DrawText(textColor, - new Point(rect.X + ColumnSpacing, y + rect.Height / 2 - ft.Bounds.Height / 2), - ft); + context.DrawText(ft, new Point(rect.X + ColumnSpacing, y + rect.Height / 2 - ft.Height / 2)); } } + } finally { @@ -241,7 +238,7 @@ private void RenderHeaders(DrawingContext context) return; FontFamily font = FontFamily.Default; - if (Application.Current!.Styles.TryGetResource("MainFontSans", out var mainFontSans) && mainFontSans is FontFamily mainFontSansFamily) + if (Application.Current!.Styles.TryGetResource("MainFontSans", SystemTheme.EffectiveThemeIsDark ? ThemeVariant.Dark : ThemeVariant.Light, out var mainFontSans) && mainFontSans is FontFamily mainFontSansFamily) font = mainFontSansFamily; var scrollViewer = ScrollViewer; @@ -265,12 +262,11 @@ private void RenderHeaders(DrawingContext context) continue; var column = Columns[i]; - var ft = new FormattedText + var ft = new FormattedText(column.Header, CultureInfo.InvariantCulture, FlowDirection.LeftToRight, + new Typeface(font, FontStyle.Normal, FontWeight.Bold), 12, BorderPen.Brush) { - Text = column.Header, - Constraint = new Size(column.Width, RowHeight), - Typeface = new Typeface(font, FontStyle.Normal, FontWeight.Bold), - FontSize = 12 + MaxTextWidth = float.MaxValue, // column.Width // we don't want text wrapping so pass float.MaxValue + MaxTextHeight = RowHeight }; bool isMouseOverColumn = isMouseOverHeader && lastMouseLocation.X >= x && lastMouseLocation.X < x + column.Width; @@ -278,7 +274,7 @@ private void RenderHeaders(DrawingContext context) context.FillRectangle(lastMouseButtonPressed && !isMouseOverSplitter ? HeaderPressedBackground : HeaderHoverBackground, new Rect(x, y, column.Width, RowHeight)); var state = context.PushClip(new Rect(x + ColumnSpacing, y, Math.Max(0, column.Width - ColumnSpacing * 2), RowHeight)); - context.DrawText(BorderPen.Brush, new Point(x + ColumnSpacing, y + RowHeight / 2 - ft.Bounds.Height / 2), ft); + context.DrawText(ft, new Point(x + ColumnSpacing, y + RowHeight / 2 - ft.Height / 2)); state.Dispose(); x += column.Width; @@ -395,4 +391,4 @@ private bool IsColumnVisible(int index) return null; } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.cs b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.cs index b3a952f7d..35d97105a 100644 --- a/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.cs +++ b/AvaloniaStyles/Controls/FastTableView/VeryFastTableView.cs @@ -5,14 +5,16 @@ using Avalonia.Controls; using Avalonia.Input; using Avalonia.Media; +using Avalonia.Styling; using Avalonia.Threading; using Avalonia.VisualTree; +using AvaloniaStyles.Controls.OptimizedVeryFastTableView; using WDE.Common.Utils; using WDE.MVVM.Observable; namespace AvaloniaStyles.Controls.FastTableView; -public partial class VeryFastTableView : Panel, IKeyboardNavigationHandler, IFastTableContext +public partial class VeryFastTableView : RenderedPanel, IFastTableContext { protected static double ColumnSpacing = 10; protected static double HeaderRowHeight = 36; @@ -36,7 +38,7 @@ public partial class VeryFastTableView : Panel, IKeyboardNavigationHandler, IFas private static bool GetResource(string key, T defaultVal, out T outT) { outT = defaultVal; - if (Application.Current!.Styles.TryGetResource(key, out var res) && res is T t) + if (Application.Current!.Styles.TryGetResource(key, SystemTheme.EffectiveThemeVariant, out var res) && res is T t) { outT = t; return true; @@ -109,10 +111,16 @@ static VeryFastTableView() table.UpdateKeyBindings(); }); } - + private Panel headersViewParent; private Panel subheadersViewParent; + private class NoArrangePanel : Panel + { + protected override Size ArrangeOverride(Size finalSize) => finalSize; + protected override Size MeasureOverride(Size availableSize) => availableSize; + } + public VeryFastTableView() { UpdateKeyBindings(); @@ -123,12 +131,47 @@ public VeryFastTableView() headerViews = new RecyclableViewList(headersViewParent); subheaderViews = new RecyclableViewList(subheadersViewParent); headersViewParent.ClipToBounds = subheadersViewParent.ClipToBounds = true; + var openAndErase = new DelegateCommand(() => + { + if (IsSelectedCellValid) + OpenEditor(""); + }); + KeyBindings.Add(new KeyBinding() + { + Gesture = new KeyGesture(Key.Enter), + Command = new DelegateCommand(() => + { + if (IsSelectedCellValid) + OpenEditor(); + }) + }); + KeyBindings.Add(new KeyBinding() + { + Gesture = new KeyGesture(Key.Back), + Command = openAndErase + }); + KeyBindings.Add(new KeyBinding() + { + Gesture = new KeyGesture(Key.Delete), + Command = openAndErase + }); } - private class NoArrangePanel : Panel + private ScrollViewer? boundScrollViewer; + + public override void ApplyTemplate() { - protected override Size ArrangeOverride(Size finalSize) => finalSize; - protected override Size MeasureOverride(Size availableSize) => availableSize; + base.ApplyTemplate(); + if (boundScrollViewer != null) + boundScrollViewer.ScrollChanged -= ScrollViewerOnScrollChanged; + if (ScrollViewer != null) + ScrollViewer.ScrollChanged += ScrollViewerOnScrollChanged; + boundScrollViewer = ScrollViewer; + } + + private void ScrollViewerOnScrollChanged(object? sender, ScrollChangedEventArgs e) + { + InvalidateVisual(); } private void RemoveInvisibleFromSelection() @@ -251,12 +294,12 @@ protected override void OnTextInput(TextInputEventArgs e) } } - protected override void OnPointerLeave(PointerEventArgs e) + protected override void OnPointerExited(PointerEventArgs e) { - Cursor = Cursor.Default; + SetCurrentValue(CursorProperty, Cursor.Default); lastMouseLocation = new Point(-1, -1); InvalidateVisual(); - base.OnPointerLeave(e); + base.OnPointerExited(e); } protected override void OnPointerMoved(PointerEventArgs e) @@ -266,7 +309,7 @@ protected override void OnPointerMoved(PointerEventArgs e) var point = currentPoint.Position; lastMouseLocation = point; lastMouseButtonPressed = currentPoint.Properties.IsLeftButtonPressed; - Cursor = (currentlyResizedColumn.HasValue || IsPointHeader(point) && IsOverColumnSplitter(lastMouseLocation.X, out _)) ? new Cursor(StandardCursorType.SizeWestEast) : Cursor.Default; + SetCurrentValue(CursorProperty, (currentlyResizedColumn.HasValue || IsPointHeader(point) && IsOverColumnSplitter(lastMouseLocation.X, out _)) ? new Cursor(StandardCursorType.SizeWestEast) : Cursor.Default); if (CustomCellDrawer != null) { @@ -307,7 +350,7 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) var index = GetRowIndexByY(e.GetPosition(this)); if (!index.HasValue || !IsRowIndexValid(index.Value)) { - SelectedRowIndex = VerticalCursor.None; + SetCurrentValue(SelectedRowIndexProperty, VerticalCursor.None); MultiSelection.Clear(); } else @@ -357,20 +400,20 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) MultiSelection.Add(index.Value); } } - SelectedRowIndex = index.Value; + SetCurrentValue(SelectedRowIndexProperty, index.Value); } } // cell { var columnIndex = GetColumnIndexByX(e.GetPosition(this).X); if (Items?.Count == 0 || !columnIndex.HasValue) - SelectedCellIndex = -1; + SetCurrentValue(SelectedCellIndexProperty, -1); else { if (TableSpan is { } span && span.IsMerged(SelectedRowIndex.GroupIndex, SelectedRowIndex.RowIndex, columnIndex.Value, out var _, out var firstColumn)) columnIndex = firstColumn; - SelectedCellIndex = Math.Clamp(columnIndex.Value, 0, ColumnsCount - 1); + SetCurrentValue(SelectedCellIndexProperty, Math.Clamp(columnIndex.Value, 0, ColumnsCount - 1)); } } } @@ -583,7 +626,7 @@ private void UpdateKeyBindings() if (IsSelectedCellValid) OpenEditor(""); }); - KeyBindings.Add(new BetterKeyBinding() + KeyBindings.Add(new VirtualizedVeryFastTableView.BetterKeyBinding() { Gesture = new KeyGesture(Key.Enter), CustomCommand = new DelegateCommand(() => @@ -592,64 +635,15 @@ private void UpdateKeyBindings() OpenEditor(); }) }); - KeyBindings.Add(new BetterKeyBinding() + KeyBindings.Add(new VirtualizedVeryFastTableView.BetterKeyBinding() { Gesture = new KeyGesture(Key.Back), CustomCommand = openAndErase }); - KeyBindings.Add(new BetterKeyBinding() + KeyBindings.Add(new VirtualizedVeryFastTableView.BetterKeyBinding() { Gesture = new KeyGesture(Key.Delete), CustomCommand = openAndErase }); } - - - /*** - * This is KeyBinding that forwards the gesture to the focused TextBox first - */ - private class BetterKeyBinding : KeyBinding, ICommand - { - public static readonly StyledProperty CustomCommandProperty = AvaloniaProperty.Register(nameof (CustomCommand)); - - public ICommand CustomCommand - { - get => GetValue(CustomCommandProperty); - set => SetValue(CustomCommandProperty, value); - } - - public BetterKeyBinding() - { - Command = this; - } - - public bool CanExecute(object? parameter) - { - if (FocusManager.Instance!.Current is TextBox tb) - return true; - return CustomCommand.CanExecute(parameter); - } - - public void Execute(object? parameter) - { - if (FocusManager.Instance!.Current is TextBox tb) - { - var ev = Activator.CreateInstance(); - ev.Key = Gesture.Key; - ev.KeyModifiers = Gesture.KeyModifiers; - ev.RoutedEvent = InputElement.KeyDownEvent; - tb.RaiseEvent(ev); - if (!ev.Handled && CanExecute(parameter)) - CustomCommand.Execute(parameter); - } - else - CustomCommand.Execute(parameter); - } - - public event EventHandler? CanExecuteChanged - { - add => CustomCommand.CanExecuteChanged += value; - remove => CustomCommand.CanExecuteChanged -= value; - } - } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/FastTableView/WdeCellDrawer.cs b/AvaloniaStyles/Controls/FastTableView/WdeCellDrawer.cs index 26b1fa9e6..6c6437fce 100644 --- a/AvaloniaStyles/Controls/FastTableView/WdeCellDrawer.cs +++ b/AvaloniaStyles/Controls/FastTableView/WdeCellDrawer.cs @@ -1,3 +1,4 @@ +using System.Globalization; using Avalonia; using Avalonia.Media; @@ -46,14 +47,11 @@ public bool Draw(DrawingContext context, IFastTableContext table, ref Rect rect, context.DrawRectangle(ButtonBackgroundPen.Brush, ButtonBorderPen, rect, 4, 4); var state = context.PushClip(rect); - var ft = new FormattedText - { - Text = "Click", - Constraint = new Size(rect.Width, rect.Height), - Typeface = Typeface.Default, - FontSize = 12 - }; - context.DrawText(ButtonTextPen.Brush, new Point(rect.Center.X - ft.Bounds.Width / 2, rect.Center.Y - ft.Bounds.Height / 2), ft); + var ft = new FormattedText("Click", CultureInfo.CurrentCulture, FlowDirection.LeftToRight, Typeface.Default, 12, ButtonTextPen.Brush); + ft.MaxTextWidth = float.MaxValue; // rect.Width // we don't want text wrapping so pass float.MaxValue + ft.MaxTextHeight = rect.Height; + + context.DrawText(ft, new Point(rect.Center.X - ft.Width / 2, rect.Center.Y - ft.Height / 2)); state.Dispose(); return true; diff --git a/AvaloniaStyles/Controls/GridView.cs b/AvaloniaStyles/Controls/GridView.cs index 8938d9fc0..fdead31fc 100644 --- a/AvaloniaStyles/Controls/GridView.cs +++ b/AvaloniaStyles/Controls/GridView.cs @@ -13,9 +13,11 @@ using Avalonia.Data; using Avalonia.Input; using Avalonia.Interactivity; +using Avalonia.LogicalTree; using Avalonia.Metadata; using Avalonia.Styling; using Avalonia.Threading; +using Avalonia.Utilities; using Avalonia.VisualTree; using WDE.MVVM; using WDE.MVVM.Observable; @@ -25,18 +27,20 @@ namespace AvaloniaStyles.Controls { public class GridViewListBox : ListBox { - protected override IItemContainerGenerator CreateItemContainerGenerator() + protected override Control CreateContainerForItemOverride(object? item, int index, object? recycleKey) { - return new ItemContainerGenerator( - this, - ContentControl.ContentProperty, - ContentControl.ContentTemplateProperty); + return new GridViewItem(); + } + + protected override bool NeedsContainerOverride(object? item, int index, out object? recycleKey) + { + return NeedsContainer(item, out recycleKey); } } - public class GridViewItem : ListBoxItem, IStyleable + public class GridViewItem : ListBoxItem { - Type IStyleable.StyleKey => typeof(ListBoxItem); + protected override Type StyleKeyOverride => typeof(ListBoxItem); static GridViewItem() { ContentProperty.Changed.AddClassHandler((item, args) => @@ -115,8 +119,8 @@ public object? SelectedItem public bool AutoScrollToSelectedItem { - get => GetValue(SelectingItemsControl.AutoScrollToSelectedItemProperty); - set => SetValue(SelectingItemsControl.AutoScrollToSelectedItemProperty, value); + get => GetValue(AutoScrollToSelectedItemProperty); + set => SetValue(AutoScrollToSelectedItemProperty, value); } public static readonly DirectProperty> ColumnsProperty = @@ -143,19 +147,29 @@ public IDataTemplate ItemTemplate public ListBox? ListBoxImpl => listBox; + private ScrollViewer? headerScroll; + private ScrollViewer? contentScroll; + protected override void OnApplyTemplate(TemplateAppliedEventArgs e) { + if (headerScroll is { }) + headerScroll.ScrollChanged -= OnHeaderScrollChanged; + if (contentScroll is { }) + contentScroll.ScrollChanged -= OnContentScrollChanged; + base.OnApplyTemplate(e); - header = e.NameScope.Find("PART_header"); - listBox = e.NameScope.Find("PART_listbox"); - + header = e.NameScope.Get("PART_header"); + listBox = e.NameScope.Get("PART_listbox"); + headerScroll = e.NameScope.Get("PART_HeaderScroll"); + contentScroll = e.NameScope.Get("PART_ContentScroll"); + + headerScroll.ScrollChanged += OnHeaderScrollChanged; + contentScroll.ScrollChanged += OnContentScrollChanged; + header.ColumnDefinitions.Clear(); header.Children.Clear(); SetupGridColumns(header, true); - // additional column in header makes it easier to resize - header.ColumnDefinitions.Add(new ColumnDefinition(20, GridUnitType.Pixel)); - int i = 0; foreach (var column in Columns) { @@ -174,17 +188,18 @@ protected override void OnApplyTemplate(TemplateAppliedEventArgs e) } BindFixMultiSelect(); - - void ExecuteScrollWhenLayoutUpdated(object? sender, EventArgs e) - { - LayoutUpdated -= ExecuteScrollWhenLayoutUpdated; - Dispatcher.UIThread.Post(AutoScrollToSelectedItemIfNecessary); - } + } - if (AutoScrollToSelectedItem) - { - LayoutUpdated += ExecuteScrollWhenLayoutUpdated; - } + private void OnContentScrollChanged(object? sender, ScrollChangedEventArgs e) + { + if (contentScroll is not null && headerScroll is not null && !MathUtilities.IsZero(e.OffsetDelta.X)) + headerScroll.Offset = headerScroll.Offset.WithX(contentScroll.Offset.X); + } + + private void OnHeaderScrollChanged(object? sender, ScrollChangedEventArgs e) + { + if (contentScroll is not null && headerScroll is not null && !MathUtilities.IsZero(e.OffsetDelta.X)) + contentScroll.Offset = contentScroll.Offset.WithX(headerScroll.Offset.X); } private void BindFixMultiSelect() @@ -207,7 +222,7 @@ private void BindFixMultiSelect() if (listBox.Selection.Count <= 0) return; - if (e.Source is not IVisual source) + if (e.Source is not Visual source) return; var point = e.GetCurrentPoint(source); @@ -215,13 +230,13 @@ private void BindFixMultiSelect() if (!point.Properties.IsLeftButtonPressed && !point.Properties.IsRightButtonPressed) return; - var containerIndex = GetContainerIndexFromEventSource(listBox, e.Source); + var containerIndex = GetContainerIndexFromEventSource(listBox, e.Source as Interactive); if (!containerIndex.HasValue) return; - var range = e.KeyModifiers.HasAllFlags(KeyModifiers.Shift); - var toggle = e.KeyModifiers.HasAllFlags(KeyModifiers.Control); + var range = e.KeyModifiers.HasFlagFast(KeyModifiers.Shift); + var toggle = e.KeyModifiers.HasFlagFast(KeyModifiers.Control); var isSelected = listBox.Selection.SelectedIndexes.Contains(containerIndex.Value); if (isSelected || toggle || range) @@ -233,7 +248,7 @@ private void BindFixMultiSelect() if (listBox.Selection.Count <= 0) return; - if (e.Source is not IVisual source) + if (e.Source is not Visual source) return; var point = e.GetCurrentPoint(source); @@ -241,7 +256,7 @@ private void BindFixMultiSelect() if (point.Properties.PointerUpdateKind == PointerUpdateKind.LeftButtonReleased || point.Properties.PointerUpdateKind == PointerUpdateKind.RightButtonReleased) { - var containerIndex = GetContainerIndexFromEventSource(listBox, e.Source); + var containerIndex = GetContainerIndexFromEventSource(listBox, e.Source as Interactive); if (containerIndex.HasValue) { @@ -253,8 +268,8 @@ private void BindFixMultiSelect() .Invoke(listBox, new object[] { containerIndex.Value, true, - e.KeyModifiers.HasAllFlags(KeyModifiers.Shift), - e.KeyModifiers.HasAllFlags(KeyModifiers.Control), + e.KeyModifiers.HasFlagFast(KeyModifiers.Shift), + e.KeyModifiers.HasFlagFast(KeyModifiers.Control), point.Properties.PointerUpdateKind == PointerUpdateKind.RightButtonReleased }); e.Handled = true; @@ -269,7 +284,6 @@ private void BindFixMultiSelect() protected override void OnAttachedToVisualTree(VisualTreeAttachmentEventArgs e) { base.OnAttachedToVisualTree(e); - AutoScrollToSelectedItemIfNecessary(); BindFixMultiSelect(); } @@ -280,13 +294,13 @@ protected override void OnDetachedFromVisualTree(VisualTreeAttachmentEventArgs e handlersDisposable = null; } - protected static int? GetContainerIndexFromEventSource(ListBox listBox, IInteractive? eventSource) + protected static int? GetContainerIndexFromEventSource(ListBox listBox, Interactive? eventSource) { - for (var current = eventSource as IVisual; current != null; current = current.VisualParent) + for (var current = eventSource as Visual; current != null; current = current.GetVisualParent()) { - if (current is IControl control && control.LogicalParent == listBox) + if (current is Control control && ReferenceEquals(control.GetLogicalParent(), listBox)) { - int? index = listBox.ItemContainerGenerator?.IndexFromContainer(control); + int? index = listBox.IndexFromContainer(control); return index == -1 ? null : index; } } @@ -314,41 +328,15 @@ private void SetupGridColumns(Grid grid, bool isHeader) grid.ColumnDefinitions.Add(c); grid.ColumnDefinitions.Add(new ColumnDefinition(SplitterWidth, GridUnitType.Pixel)); } - } - - private void AutoScrollToSelectedItemIfNecessary() - { - if (listBox == null) - return; - - if (!AutoScrollToSelectedItem) - return; - - if (SelectedItem == null) - return; - var index = listBox.SelectedIndex; - - var visible = listBox.GetVisualDescendants().Count(t => t is ListBoxItem); - - var scroll = listBox.FindDescendantOfType(); - - if (index < scroll.Offset.Y || index > scroll.Offset.Y + visible) - scroll.Offset = new Vector(scroll.Offset.X, Math.Max(0, index - visible / 2)); + // additional column in header makes it easier to resize + grid.ColumnDefinitions.Add(new ColumnDefinition(20, GridUnitType.Pixel)); } static GridView() { ColumnsProperty.Changed.AddClassHandler(OnColumnsModified); - SelectedItemProperty.Changed.AddClassHandler((view, args) => - { - if (!view.AutoScrollToSelectedItem) - return; - - Dispatcher.UIThread.Post(view.AutoScrollToSelectedItemIfNecessary); - }); - SelectionProperty.Changed.AddClassHandler((view, args) => { view.selectionDisposable?.Dispose(); @@ -380,7 +368,7 @@ protected override void OnKeyDown(KeyEventArgs e) if (list?.SelectedItem == null) return; - var selected = list.ItemContainerGenerator.ContainerFromIndex(list.SelectedIndex); + var selected = list.ContainerFromIndex(list.SelectedIndex); if (selected == null) return; @@ -397,17 +385,17 @@ protected override void OnKeyDown(KeyEventArgs e) public GridView() { - Selection = new SelectionModel(); - ItemTemplate = new FuncDataTemplate(_ => true, (_, _) => + SetCurrentValue(SelectionProperty, new SelectionModel()); + SetCurrentValue(ItemTemplateProperty, new FuncDataTemplate(_ => true, (_, _) => { var parent = ConstructGrid(); int i = 0; foreach (var column in Columns) { - IControl control; + Control control; if (column.DataTemplate is { } dt) { - control = dt.Build(column); + control = dt.Build(column) ?? new TextBlock(){Text = "Couldn't instantiate column"}; } else if (column.Checkable) { @@ -430,7 +418,7 @@ public GridView() parent.Children.Add(control); } return parent; - }, true); + }, true)); } } diff --git a/AvaloniaStyles/Controls/GridViewStyle.axaml b/AvaloniaStyles/Controls/GridViewStyle.axaml index 94019e817..9c7954317 100644 --- a/AvaloniaStyles/Controls/GridViewStyle.axaml +++ b/AvaloniaStyles/Controls/GridViewStyle.axaml @@ -1,36 +1,13 @@ + xmlns:controls="clr-namespace:AvaloniaStyles.Controls" + xmlns:converters="clr-namespace:AvaloniaStyles.Converters"> - - @@ -64,56 +33,32 @@ - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + diff --git a/AvaloniaStyles/Controls/GroupingListBox.axaml b/AvaloniaStyles/Controls/GroupingListBox.axaml index 1b1b5dcb4..2781437fb 100644 --- a/AvaloniaStyles/Controls/GroupingListBox.axaml +++ b/AvaloniaStyles/Controls/GroupingListBox.axaml @@ -35,16 +35,14 @@ BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}"> - + - + @@ -61,11 +59,8 @@ + Margin="{TemplateBinding Padding}"/> diff --git a/AvaloniaStyles/Controls/GroupingListBox.axaml.cs b/AvaloniaStyles/Controls/GroupingListBox.axaml.cs index 1cc03f621..a1291dde1 100644 --- a/AvaloniaStyles/Controls/GroupingListBox.axaml.cs +++ b/AvaloniaStyles/Controls/GroupingListBox.axaml.cs @@ -1,5 +1,6 @@ using System; using System.Collections; +using System.Linq; using Avalonia; using Avalonia.Collections; using Avalonia.Controls; @@ -8,6 +9,7 @@ using Avalonia.Data; using Avalonia.Input; using Avalonia.Interactivity; +using Avalonia.LogicalTree; using Avalonia.Metadata; using Avalonia.Styling; using Avalonia.VisualTree; @@ -31,41 +33,41 @@ public object? GroupName set => SetAndRaise(GroupNameProperty, ref groupName, value); } - public static readonly DirectProperty CustomContentProperty = - AvaloniaProperty.RegisterDirect( + public static readonly DirectProperty CustomContentProperty = + AvaloniaProperty.RegisterDirect( nameof(CustomContent), o => o.CustomContent, (o, v) => o.CustomContent = v); - private IControl? customContent; - public IControl? CustomContent + private Control? customContent; + public Control? CustomContent { get => customContent; set => SetAndRaise(CustomContentProperty, ref customContent, value); } - public static readonly DirectProperty CustomRightContentProperty = - AvaloniaProperty.RegisterDirect( + public static readonly DirectProperty CustomRightContentProperty = + AvaloniaProperty.RegisterDirect( nameof(CustomRightContent), o => o.CustomRightContent, (o, v) => o.CustomRightContent = v); - private IControl? customRightContent; - public IControl? CustomRightContent + private Control? customRightContent; + public Control? CustomRightContent { get => customRightContent; set => SetAndRaise(CustomRightContentProperty, ref customRightContent, value); } - public static readonly DirectProperty CustomCenterContentProperty = - AvaloniaProperty.RegisterDirect( + public static readonly DirectProperty CustomCenterContentProperty = + AvaloniaProperty.RegisterDirect( nameof(CustomCenterContent), o => o.CustomCenterContent, (o, v) => o.CustomCenterContent = v); - private IControl? customCenterContent; - public IControl? CustomCenterContent + private Control? customCenterContent; + public Control? CustomCenterContent { get => customCenterContent; set => SetAndRaise(CustomCenterContentProperty, ref customCenterContent, value); @@ -106,16 +108,16 @@ public IEnumerable Items set => SetAndRaise(ItemsProperty, ref items, value); } - private static readonly FuncTemplate DefaultPanel = - new FuncTemplate(() => new StackPanel()); + private static readonly FuncTemplate DefaultPanel = + new FuncTemplate(() => new StackPanel()); - public static readonly StyledProperty> ItemsPanelProperty = - AvaloniaProperty.Register>(nameof(ItemsPanel), DefaultPanel); + public static readonly StyledProperty> ItemsPanelProperty = + AvaloniaProperty.Register>(nameof(ItemsPanel), DefaultPanel); public static readonly StyledProperty ItemTemplateProperty = AvaloniaProperty.Register(nameof(ItemTemplate)); - public ITemplate ItemsPanel + public ITemplate ItemsPanel { get => GetValue(ItemsPanelProperty); set => SetValue(ItemsPanelProperty, value); @@ -153,8 +155,9 @@ public void ScrollToItem(object item) if (index == -1) return; - var container = parentItems.ItemContainerGenerator.ContainerFromIndex(index); - scroll.Offset = new Vector(0, container.Bounds.Y); + var container = parentItems.ContainerFromIndex(index); + if (container != null) + scroll.Offset = new Vector(0, container.Bounds.Y); } protected override void OnApplyTemplate(TemplateAppliedEventArgs e) @@ -180,7 +183,7 @@ public void FocusElement(int parentIndex, int innerIndex) return; listBox.SelectedIndex = innerIndex; - (listBox.ItemContainerGenerator.ContainerFromIndex(listBox.SelectedIndex)).Focus(); + (listBox.ContainerFromIndex(listBox.SelectedIndex))?.Focus(); } #region KEYBOARD_NAVIGATION @@ -207,14 +210,17 @@ private void OnPreviewKeyDown(object? sender, KeyEventArgs e) { e.Handled = true; var dir = e.Key == Key.Up ? -1 : 1; - int itemsInRow = (int)(that.Bounds.Width / ((that.ItemContainerGenerator.ContainerFromIndex(0)).Bounds.Width)); + var element = that.ContainerFromIndex(0); + if (element == null) + return; + int itemsInRow = (int)(that.Bounds.Width / (element.Bounds.Width)); MoveUpDown(that, parent, itemsInRow, dir); } } private ListBox? GetNextParent(ItemsControl parent, ListBox current, int dir) { - var currentParentIndex = ((IList)parent.Items).IndexOf(current.DataContext); + var currentParentIndex = ((IList?)parent.Items)?.IndexOf(current.DataContext) ?? -1; var nextParentIndex = currentParentIndex + dir; if (nextParentIndex < 0 || nextParentIndex > parent.ItemCount - 1) return null; @@ -223,10 +229,13 @@ private void OnPreviewKeyDown(object? sender, KeyEventArgs e) private ListBox? GetListBoxFromItemsControl(ItemsControl parent, int index) { - var x = parent.ItemContainerGenerator.ContainerFromIndex(index); - if (x == null || x.LogicalChildren.Count == 0 || x.LogicalChildren[0].LogicalChildren.Count < 2) + var x = parent.ContainerFromIndex(index); + if (x == null) + return null; + var logicalChildren = x.GetLogicalChildren().ToList(); + if (x == null || logicalChildren.Count == 0 || logicalChildren[0].LogicalChildren.Count < 2) return null; - return (ListBox) x.LogicalChildren[0].LogicalChildren[1]; + return (ListBox) logicalChildren[0].LogicalChildren[1]; } private void MoveNextPrev(ListBox listBox, ItemsControl parent, int dir) @@ -294,17 +303,17 @@ private int FindIndexInColumn(int startIndex, int length, int column, int column private void SelectIndex(ListBox listbox, int index) { listbox.SelectedIndex = index; - (listbox.ItemContainerGenerator.ContainerFromIndex(listbox.SelectedIndex)).Focus(); + listbox.ContainerFromIndex(listbox.SelectedIndex)?.Focus(); } - public static T? FindParent(IVisual? obj) where T : Control + public static T? FindParent(Visual? obj) where T : Control { - obj = obj?.VisualParent; + obj = obj?.GetVisualParent(); while (obj != null) { if (obj is T parent) return parent; - obj = obj.VisualParent; + obj = obj.GetVisualParent(); } return null; @@ -312,9 +321,9 @@ private void SelectIndex(ListBox listbox, int index) #endregion } - internal class GroupingListBoxInner : ListBox, IStyleable + internal class GroupingListBoxInner : ListBox { - Type IStyleable.StyleKey => typeof(ListBox); + protected override Type StyleKeyOverride => typeof(ListBox); public static readonly DirectProperty CustomSelectedItemProperty = AvaloniaProperty.RegisterDirect( @@ -352,7 +361,7 @@ static GroupingListBoxInner() inEvent = true; { o.CustomSelectedItem = arg.NewValue; - GroupingListBox parent = o.FindAncestorOfType(); + GroupingListBox? parent = o.FindAncestorOfType(); if (parent != null) parent.SelectedItem = arg.NewValue; o.RaisePropertyChanged(CustomSelectedItemProperty, arg.OldValue, arg.NewValue); @@ -362,6 +371,6 @@ static GroupingListBoxInner() }); } - private bool IsAttached() => ((IVisual) this).IsAttachedToVisualTree; + private bool IsAttached() => ((Visual) this).GetVisualRoot() != null; } } \ No newline at end of file diff --git a/AvaloniaStyles/Controls/HamburgerMenuButton.axaml b/AvaloniaStyles/Controls/HamburgerMenuButton.axaml new file mode 100644 index 000000000..e04e1916c --- /dev/null +++ b/AvaloniaStyles/Controls/HamburgerMenuButton.axaml @@ -0,0 +1,55 @@ + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/AvaloniaStyles/Controls/HamburgerMenuButton.axaml.cs b/AvaloniaStyles/Controls/HamburgerMenuButton.axaml.cs new file mode 100644 index 000000000..4639c8d46 --- /dev/null +++ b/AvaloniaStyles/Controls/HamburgerMenuButton.axaml.cs @@ -0,0 +1,9 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Controls.Primitives; + +namespace AvaloniaStyles.Controls; + +public class HamburgerMenuButton : TemplatedControl +{ +} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/IconElement/BitmapIcon.cs b/AvaloniaStyles/Controls/IconElement/BitmapIcon.cs index afa82c513..622e04b3a 100644 --- a/AvaloniaStyles/Controls/IconElement/BitmapIcon.cs +++ b/AvaloniaStyles/Controls/IconElement/BitmapIcon.cs @@ -6,19 +6,24 @@ using Avalonia.Skia; using SkiaSharp; using System; -using Avalonia.Visuals.Media.Imaging; namespace AvaloniaStyles.Controls; public partial class BitmapIcon : FAIconElement { + public BitmapIcon() + { + RenderOptions.SetBitmapInterpolationMode(this, BitmapInterpolationMode.HighQuality); + } + ~BitmapIcon() { Dispose(); UnlinkFromBitmapIconSource(); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + /// + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == UriSourceProperty) @@ -26,7 +31,7 @@ protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs if (_bis != null) throw new InvalidOperationException("Cannot edit properties of BitmapIcon if BitmapIconSource is linked"); - CreateBitmap(UriSource); + CreateBitmap(change.GetNewValue()); InvalidateVisual(); } else if (change.Property == ShowAsMonochromeProperty) @@ -38,6 +43,7 @@ protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs } } + /// protected override Size MeasureOverride(Size availableSize) { if (_bis != null) @@ -49,6 +55,7 @@ protected override Size MeasureOverride(Size availableSize) return _originalSize; } + /// public override void Render(DrawingContext context) { if (_bitmap == null && _bis == null) @@ -64,42 +71,43 @@ public override void Render(DrawingContext context) var wid = (int)dst.Width; var hei = (int)dst.Height; - using (var bmp = new RenderTargetBitmap(new PixelSize(wid, hei))) - using (var ctx = bmp.PlatformImpl.Item.CreateDrawingContext(null)) + using (var bmp = new WriteableBitmap(new PixelSize(wid, hei), new Vector(96, 96), + PixelFormats.Bgra8888, AlphaFormat.Premul)) { - throw new NotImplementedException(); // todo Avalonia 11 - //var feat = ctx.GetFeature(); - //if (feat == null) - // throw new Exception("BitmapIcon requires SkiaSharp to be rendering backend"); - //using var lease = feat.Lease(); - //var skDC = lease.SkCanvas; -// - //skDC.Clear(new SKColor(0, 0, 0, 0)); -// - //var finalBmp = _bitmap.Resize(new SKImageInfo(wid, hei), SKFilterQuality.High); -// - //if (ShowAsMonochrome) - //{ - // var avColor = Foreground is ISolidColorBrush sc ? sc.Color : Colors.White; -// - // var color = new SKColor(avColor.R, avColor.G, avColor.B, avColor.A); - // SKPaint paint = new SKPaint(); - // paint.ColorFilter = SKColorFilter.CreateBlendMode(color, SKBlendMode.SrcATop); -// - // skDC.DrawBitmap(finalBmp, new SKRect(0, 0, (float)wid, (float)hei), paint); - // paint.Dispose(); - //} - //else - //{ - // skDC.DrawBitmap(finalBmp, new SKRect(0, 0, (float)wid, (float)hei)); - //} -// - //finalBmp.Dispose(); -// - //using (context.PushClip(dst)) - //{ - // context.DrawImage(bmp, new Rect(bmp.Size), dst, BitmapInterpolationMode.HighQuality); - //} + using var buffer = bmp.Lock(); + + var skSfc = SKSurface.Create(new SKImageInfo(wid, hei), buffer.Address); + if (skSfc == null) + return; + + var skDC = skSfc.Canvas; + + skDC.Clear(new SKColor(0, 0, 0, 0)); + + var finalBmp = _bitmap.Resize(new SKImageInfo(wid, hei), SKFilterQuality.High); + + if (ShowAsMonochrome) + { + var avColor = Foreground is ISolidColorBrush sc ? sc.Color : Colors.White; + + var color = new SKColor(avColor.R, avColor.G, avColor.B, avColor.A); + SKPaint paint = new SKPaint(); + paint.ColorFilter = SKColorFilter.CreateBlendMode(color, SKBlendMode.SrcATop); + + skDC.DrawBitmap(finalBmp, new SKRect(0, 0, (float)wid, (float)hei), paint); + paint.Dispose(); + } + else + { + skDC.DrawBitmap(finalBmp, new SKRect(0, 0, (float)wid, (float)hei)); + } + + finalBmp.Dispose(); + + using (context.PushClip(dst)) + { + context.DrawImage(bmp, new Rect(bmp.Size), dst); + } } } @@ -119,12 +127,12 @@ private void CreateBitmap(Uri src) } else { - var assets = AvaloniaLocator.Current.GetService(); - _bitmap = SKBitmap.Decode(assets.Open(src)); + _bitmap = SKBitmap.Decode(AssetLoader.Open(src)); } _originalSize = new Size(_bitmap.Width, _bitmap.Height); } + /// protected void Dispose() { _bitmap?.Dispose(); diff --git a/AvaloniaStyles/Controls/IconElement/BitmapIconSource.cs b/AvaloniaStyles/Controls/IconElement/BitmapIconSource.cs index 189aa0ce6..c3cb64b06 100644 --- a/AvaloniaStyles/Controls/IconElement/BitmapIconSource.cs +++ b/AvaloniaStyles/Controls/IconElement/BitmapIconSource.cs @@ -50,13 +50,13 @@ public bool ShowAsMonochrome public event EventHandler OnBitmapChanged; - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == UriSourceProperty) { - CreateBitmap(UriSource); + CreateBitmap(change.GetNewValue()); } else if (change.Property == ShowAsMonochromeProperty) { @@ -87,8 +87,7 @@ private void CreateBitmap(Uri src) } else { - var assets = AvaloniaLocator.Current.GetService(); - _bitmap = SKBitmap.Decode(assets.Open(src)); + _bitmap = SKBitmap.Decode(AssetLoader.Open(src)); } _originalSize = new Size(_bitmap.Width, _bitmap.Height); diff --git a/AvaloniaStyles/Controls/IconElement/FAIconElement.cs b/AvaloniaStyles/Controls/IconElement/FAIconElement.cs index 8c4b15393..8be920b25 100644 --- a/AvaloniaStyles/Controls/IconElement/FAIconElement.cs +++ b/AvaloniaStyles/Controls/IconElement/FAIconElement.cs @@ -5,6 +5,7 @@ using System; using System.ComponentModel; using System.Globalization; +using Avalonia.Controls.Documents; namespace AvaloniaStyles.Controls; @@ -18,7 +19,7 @@ public class FAIconElement : Control /// Defines the property /// public static readonly AttachedProperty ForegroundProperty = - TextBlock.ForegroundProperty.AddOwner(); + TextElement.ForegroundProperty.AddOwner(); /// /// Gets or sets a brush that describes the foreground color. @@ -29,7 +30,7 @@ public IBrush Foreground set => SetValue(ForegroundProperty, value); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); diff --git a/AvaloniaStyles/Controls/IconElement/FAPathIcon.cs b/AvaloniaStyles/Controls/IconElement/FAPathIcon.cs index 579c5e449..9e4ebe8c1 100644 --- a/AvaloniaStyles/Controls/IconElement/FAPathIcon.cs +++ b/AvaloniaStyles/Controls/IconElement/FAPathIcon.cs @@ -18,7 +18,7 @@ static FAPathIcon() ClipToBoundsProperty.OverrideDefaultValue(true); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); @@ -187,7 +187,7 @@ public override void Render(DrawingContext context) if (geometry == null) return; - using var s = context.PushPostTransform(_transform); // todo: post transsform? + using var s = context.PushTransform(_transform); context.DrawGeometry(Foreground, null, geometry); } diff --git a/AvaloniaStyles/Controls/IconElement/FAPathIcon.properties.cs b/AvaloniaStyles/Controls/IconElement/FAPathIcon.properties.cs index ef21f5f3b..7efe4e281 100644 --- a/AvaloniaStyles/Controls/IconElement/FAPathIcon.properties.cs +++ b/AvaloniaStyles/Controls/IconElement/FAPathIcon.properties.cs @@ -15,7 +15,7 @@ public partial class FAPathIcon : FAIconElement /// /// Defines the property /// - public static StyledProperty DataProperty = + public static readonly StyledProperty DataProperty = Path.DataProperty.AddOwner(); /// diff --git a/AvaloniaStyles/Controls/IconElement/FontIcon.cs b/AvaloniaStyles/Controls/IconElement/FontIcon.cs index 9fa80041b..55b198100 100644 --- a/AvaloniaStyles/Controls/IconElement/FontIcon.cs +++ b/AvaloniaStyles/Controls/IconElement/FontIcon.cs @@ -2,6 +2,7 @@ using System.Diagnostics; using Avalonia; using Avalonia.Controls; +using Avalonia.Controls.Documents; using Avalonia.Media; using Avalonia.Media.TextFormatting; using Avalonia.Threading; @@ -14,20 +15,20 @@ namespace AvaloniaStyles.Controls; /// public partial class FontIcon : FAIconElement { - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); - if (change.Property == TextBlock.FontSizeProperty || - change.Property == TextBlock.FontFamilyProperty || - change.Property == TextBlock.FontWeightProperty || - change.Property == TextBlock.FontStyleProperty || + if (change.Property == TextElement.FontSizeProperty || + change.Property == TextElement.FontFamilyProperty || + change.Property == TextElement.FontWeightProperty || + change.Property == TextElement.FontStyleProperty || change.Property == GlyphProperty) { _textLayout = null; InvalidateMeasure(); } - else if (change.Property == TextBlock.ForegroundProperty) + else if (change.Property == TextElement.ForegroundProperty) { _textLayout = null; // FAIconElement calls InvalidateVisual @@ -41,7 +42,7 @@ protected override Size MeasureOverride(Size availableSize) GenerateText(); } - return _textLayout.Size; + return new Size(_textLayout.Width, _textLayout.Height); } public override void Render(DrawingContext context) @@ -52,10 +53,9 @@ public override void Render(DrawingContext context) var dstRect = new Rect(Bounds.Size); using (context.PushClip(dstRect)) { - var pt = new Point(dstRect.Center.X - _textLayout.Size.Width / 2, - dstRect.Center.Y - _textLayout.Size.Height / 2); - using var _ = context.PushPreTransform(Matrix.CreateTranslation(pt)); - _textLayout.Draw(context); // , pt todo Avalonia 11 + var pt = new Point(dstRect.Center.X - _textLayout.Width / 2, + dstRect.Center.Y - _textLayout.Height / 2); + _textLayout.Draw(context, pt); } } diff --git a/AvaloniaStyles/Controls/IconElement/IconSourceElement.cs b/AvaloniaStyles/Controls/IconElement/IconSourceElement.cs index 9ef85c548..759f23703 100644 --- a/AvaloniaStyles/Controls/IconElement/IconSourceElement.cs +++ b/AvaloniaStyles/Controls/IconElement/IconSourceElement.cs @@ -25,7 +25,7 @@ public IconSource IconSource set => SetValue(IconSourceProperty, value); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); diff --git a/AvaloniaStyles/Controls/IconElement/ImageIcon.cs b/AvaloniaStyles/Controls/IconElement/ImageIcon.cs index d4c347b34..60141d64b 100644 --- a/AvaloniaStyles/Controls/IconElement/ImageIcon.cs +++ b/AvaloniaStyles/Controls/IconElement/ImageIcon.cs @@ -2,7 +2,6 @@ using Avalonia.Media; using Avalonia.Media.Imaging; using Avalonia.Metadata; -using Avalonia.Visuals.Media.Imaging; namespace AvaloniaStyles.Controls; @@ -27,7 +26,7 @@ public IImage Source set => SetValue(SourceProperty, value); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == SourceProperty) @@ -59,7 +58,7 @@ public override void Render(DrawingContext context) Rect srcRect = new Rect(src.Size) .CenterRect(new Rect(destRect.Size / scale)); - context.DrawImage(src, srcRect, destRect, BitmapInterpolationMode.HighQuality); + context.DrawImage(src, srcRect, destRect); } } } diff --git a/AvaloniaStyles/Controls/IconElement/PathIconSource.cs b/AvaloniaStyles/Controls/IconElement/PathIconSource.cs index d37ede9a6..e3d6fd74c 100644 --- a/AvaloniaStyles/Controls/IconElement/PathIconSource.cs +++ b/AvaloniaStyles/Controls/IconElement/PathIconSource.cs @@ -17,7 +17,7 @@ static PathIconSource() /// /// Defines the property /// - public static StyledProperty DataProperty = + public static readonly StyledProperty DataProperty = FAPathIcon.DataProperty.AddOwner(); /// @@ -34,7 +34,7 @@ public Geometry Data /// Defines the property. /// public static readonly StyledProperty StretchProperty = - FAPathIcon.StretchProperty.AddOwner(); + FAPathIcon.StretchProperty.AddOwner(); /// /// Gets or sets a enumeration value that describes how the shape fills its allocated space. @@ -49,7 +49,7 @@ public Stretch Stretch /// Defines the property. /// public static readonly StyledProperty StretchDirectionProperty = - FAPathIcon.StretchDirectionProperty.AddOwner(); + FAPathIcon.StretchDirectionProperty.AddOwner(); /// /// Gets or sets a value controlling in what direction contents will be stretched. diff --git a/AvaloniaStyles/Controls/IconElement/SymbolIcon.cs b/AvaloniaStyles/Controls/IconElement/SymbolIcon.cs index 644d9ded5..3e1e9b80c 100644 --- a/AvaloniaStyles/Controls/IconElement/SymbolIcon.cs +++ b/AvaloniaStyles/Controls/IconElement/SymbolIcon.cs @@ -47,7 +47,7 @@ public double FontSize set => SetValue(FontSizeProperty, value); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == TextBlock.FontSizeProperty || @@ -77,7 +77,7 @@ protected override Size MeasureOverride(Size availableSize) if (_textLayout == null) GenerateText(); - return _textLayout?.Size ?? default; + return new Size(_textLayout.Width, _textLayout.Height); } public override void Render(DrawingContext context) @@ -88,10 +88,9 @@ public override void Render(DrawingContext context) var dstRect = new Rect(Bounds.Size); using (context.PushClip(dstRect)) { - var pt = new Point(dstRect.Center.X - _textLayout.Size.Width / 2, - dstRect.Center.Y - _textLayout.Size.Height / 2); - using var _ = context.PushPostTransform(Matrix.CreateTranslation(pt)); - _textLayout.Draw(context); // , pt todo Avalonia 11 + var pt = new Point(dstRect.Center.X - _textLayout.Width / 2, + dstRect.Center.Y - _textLayout.Height / 2); + _textLayout.Draw(context, pt); } } diff --git a/AvaloniaStyles/Controls/InfoBar/InfoBar.cs b/AvaloniaStyles/Controls/InfoBar/InfoBar.cs index c8dbe59fe..7cf0e9a83 100644 --- a/AvaloniaStyles/Controls/InfoBar/InfoBar.cs +++ b/AvaloniaStyles/Controls/InfoBar/InfoBar.cs @@ -39,7 +39,7 @@ protected override void OnApplyTemplate(TemplateAppliedEventArgs e) UpdateForeground(); } - protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) + protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == IsOpenProperty) @@ -77,7 +77,7 @@ protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs } } - protected override bool RegisterContentPresenter(IContentPresenter presenter) + protected override bool RegisterContentPresenter(ContentPresenter presenter) { if (presenter.Name == "ContentPresenter") return true; @@ -89,7 +89,7 @@ private void OnCloseButtonClick(object? sender, RoutedEventArgs e) { CloseButtonClick?.Invoke(this, EventArgs.Empty); _lastCloseReason = InfoBarCloseReason.CloseButton; - IsOpen = false; + SetCurrentValue(IsOpenProperty, false); } private void RaiseClosingEvent() @@ -107,7 +107,7 @@ private void RaiseClosingEvent() { // The developer has changed the Cancel property to true, // so we need to revert the IsOpen property to true. - IsOpen = true; + SetCurrentValue(IsOpenProperty, true); } } diff --git a/AvaloniaStyles/Controls/LightSolidColorBrush.cs b/AvaloniaStyles/Controls/LightSolidColorBrush.cs deleted file mode 100644 index 486e14562..000000000 --- a/AvaloniaStyles/Controls/LightSolidColorBrush.cs +++ /dev/null @@ -1,42 +0,0 @@ -using Avalonia; -using Avalonia.Media; -using Avalonia.Media.Immutable; -using Avalonia.Metadata; -using AvaloniaStyles.Utils; - -namespace AvaloniaStyles.Controls; - -public class LightSolidColorBrush : Brush, ISolidColorBrush -{ - public override IBrush ToImmutable() => new ImmutableSolidColorBrush(this); - - public static readonly StyledProperty BaseProperty = - AvaloniaProperty.Register(nameof(Base)); - - public static readonly StyledProperty LightProperty = - AvaloniaProperty.Register(nameof(Light)); - - [Content] - public ISolidColorBrush Base - { - get => GetValue(BaseProperty); - set => SetValue(BaseProperty, value); - } - - public double Light - { - get => GetValue(LightProperty); - set => SetValue(LightProperty, value); - } - - public Color Color - { - get - { - var baseBrush = Base; - var hsl = HslColor.FromRgba(baseBrush.Color); - hsl = hsl.WithLightness(hsl.V + Light); - return hsl.ToRgba(baseBrush.Opacity); - } - } -} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/ManagedMenu.cs b/AvaloniaStyles/Controls/ManagedMenu.cs deleted file mode 100644 index 12278f3c1..000000000 --- a/AvaloniaStyles/Controls/ManagedMenu.cs +++ /dev/null @@ -1,23 +0,0 @@ -using Avalonia; -using Avalonia.Controls; - -namespace AvaloniaStyles.Controls; - -/// -/// for some reason, .net 7.0 and this old Avalonia version doesn't like the NativeMenu class -/// probably in Avalonia 11 it is fixed, but for now I've disabled NativeMenu -/// for Ava11, remove this and change all ManagedMenu references to NativeMenu -/// -public class ManagedMenu -{ - public static readonly AttachedProperty MenuProperty - = AvaloniaProperty.RegisterAttached("Menu"); - - public static void SetMenu(AvaloniaObject o, NativeMenu menu) => o.SetValue(MenuProperty, menu); - - public static NativeMenu GetMenu(AvaloniaObject o) => o.GetValue(MenuProperty); - - static ManagedMenu() - { - } -} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/NumberIndicatorControl/NumberIndicator.axaml b/AvaloniaStyles/Controls/NumberIndicatorControl/NumberIndicator.axaml index 9857c29cd..d1481f965 100644 --- a/AvaloniaStyles/Controls/NumberIndicatorControl/NumberIndicator.axaml +++ b/AvaloniaStyles/Controls/NumberIndicatorControl/NumberIndicator.axaml @@ -10,7 +10,7 @@ - + str) - { - var len = str.Length; - for (int i = 0; i < len; ++i) - { - if (str[i] >= 0x3FF) - return true; + LOG.LogError(e); } - - return false; } public void AutoFitColumnsWidth() @@ -88,15 +60,15 @@ public void AutoFitColumnsWidth() return; var controller = Controller; - GetFonts(out var defaultFont, out _); + var defaultFont = TextElement.GetFontFamily(this); var defaultTypeface = new Typeface(defaultFont); var boldTypeface = new Typeface(defaultFont,default, FontWeight.Bold); Span columnWidths = stackalloc double[columns.Count]; var (firstVisibleIndex, lastVisibleIndex) = GetFirstAndLastVisibleRows(); for (int i = 0; i < columns.Count; i++) { - var ft = new FormattedText(columns[i].Header, boldTypeface, 12, TextAlignment.Left, TextWrapping.NoWrap, new Size(100000, 10000)); - columnWidths[i] = ft.Bounds.Width + ColumnSpacing * 2 + 20; // this +20 is only for SqlEditor which displays a key icon for primary key columns. + var ft = new FormattedText(columns[i].Header, CultureInfo.InvariantCulture, FlowDirection.LeftToRight, boldTypeface, 12, Brushes.Black); + columnWidths[i] = ft.Width + ColumnSpacing * 2 + 20; // this +20 is only for SqlEditor which displays a key icon for primary key columns. // If this control is reused for another editor, then it might be changed, but still pls keep // the space in case of SqlEditor. } @@ -106,8 +78,8 @@ public void AutoFitColumnsWidth() for (int cellIndex = 0; cellIndex < columnWidths.Length; ++cellIndex) { var cellText = controller.GetCellText(rowIndex, cellIndex); - var ft = new FormattedText(cellText, defaultTypeface, 12, TextAlignment.Left, TextWrapping.NoWrap, new Size(100000, 10000)); - columnWidths[cellIndex] = Math.Max(columnWidths[cellIndex], ft.Bounds.Width + ColumnSpacing * 2); + var ft = new FormattedText(cellText ?? "", CultureInfo.InvariantCulture, FlowDirection.LeftToRight, defaultTypeface, 12, Brushes.Black); + columnWidths[cellIndex] = Math.Max(columnWidths[cellIndex], ft.Width + ColumnSpacing * 2); } } @@ -132,8 +104,8 @@ public void AutoFitColumnsWidth() private void RenderImpl(DrawingContext context) { - GetTypefaces(out var font, out var unicodeFallback); - + var font = new Typeface(TextElement.GetFontFamily(this)); + var actualWidth = Bounds.Width; var viewPort = DataViewport; @@ -207,22 +179,18 @@ private void RenderImpl(DrawingContext context) text = text.Substring(0, indexOfEndOfLine); rect = rect.WithWidth(rect.Width - ColumnSpacing); - var ft = new FormattedText + var ft = new FormattedText(text, CultureInfo.InvariantCulture, FlowDirection.LeftToRight, font, 12, textColor) { - Text = text, - Constraint = new Size(rect.Width, RowHeight), - Typeface = UseFallbackUnicodeFont(text) ? unicodeFallback : font, - FontSize = 12 + MaxTextHeight = RowHeight, + MaxTextWidth = float.MaxValue // rect.Width // we don't want text wrapping so pass float.MaxValue }; + if (Math.Abs(rectWidth - rect.Width) > 0.01) { state.Dispose(); state = context.PushClip(rect); } - - context.DrawText(textColor, - new Point(rect.X + ColumnSpacing, y + RowHeight / 2 - ft.Bounds.Height / 2), - ft); + context.DrawText(ft, new Point(rect.X + ColumnSpacing, rect.Center.Y - ft.Height / 2)); } } } @@ -257,7 +225,7 @@ private void RenderHeaders(DrawingContext context) var controller = Controller; FontFamily font = FontFamily.Default; - if (Application.Current!.Styles.TryGetResource("MainFontSans", out var mainFontSans) && mainFontSans is FontFamily mainFontSansFamily) + if (Application.Current!.Styles.TryGetResource("MainFontSans", SystemTheme.EffectiveThemeVariant, out var mainFontSans) && mainFontSans is FontFamily mainFontSansFamily) font = mainFontSansFamily; var scrollViewer = ScrollViewer; @@ -281,12 +249,10 @@ private void RenderHeaders(DrawingContext context) continue; var column = Columns[i]; - var ft = new FormattedText + var ft = new FormattedText(column.Header, CultureInfo.InvariantCulture, FlowDirection.LeftToRight, new Typeface(font, FontStyle.Normal, FontWeight.Bold), 12, BorderPen.Brush) { - Text = column.Header, - Constraint = new Size(column.Width, RowHeight), - Typeface = new Typeface(font, FontStyle.Normal, FontWeight.Bold), - FontSize = 12 + MaxTextHeight = RowHeight, + MaxTextWidth = float.MaxValue // column.Width // we don't want text wrapping so pass float.MaxValue }; bool isMouseOverColumn = isMouseOverHeader && lastMouseLocation.X >= x && lastMouseLocation.X < x + column.Width; @@ -296,7 +262,7 @@ private void RenderHeaders(DrawingContext context) var cellRect = new Rect(x + ColumnSpacing, y, Math.Max(0, column.Width - ColumnSpacing * 2), RowHeight); controller.DrawHeader(i, context, this, ref cellRect); var state = context.PushClip(cellRect); - context.DrawText(BorderPen.Brush, new Point(cellRect.X, cellRect.Center.Y - ft.Bounds.Height / 2), ft); + context.DrawText(ft, new Point(cellRect.X, cellRect.Center.Y - ft.Height / 2)); state.Dispose(); x += column.Width; @@ -357,4 +323,4 @@ private int GetRowIndexByY(double mouseY) return index; return -1; } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/OptimizedVeryFastTableView/VirtualizedVeryFastTableView.cs b/AvaloniaStyles/Controls/OptimizedVeryFastTableView/VirtualizedVeryFastTableView.cs index 6486f12ba..6baa971b4 100644 --- a/AvaloniaStyles/Controls/OptimizedVeryFastTableView/VirtualizedVeryFastTableView.cs +++ b/AvaloniaStyles/Controls/OptimizedVeryFastTableView/VirtualizedVeryFastTableView.cs @@ -3,17 +3,21 @@ using System.Windows.Input; using Avalonia; using Avalonia.Controls; +using Avalonia.Controls.ApplicationLifetimes; using Avalonia.Input; using Avalonia.Media; +using Avalonia.Styling; using Avalonia.Threading; using Avalonia.VisualTree; +using AvaloniaEdit; +using AvaloniaEdit.Editing; using AvaloniaStyles.Controls.FastTableView; using WDE.Common.Utils; using WDE.MVVM.Observable; namespace AvaloniaStyles.Controls.OptimizedVeryFastTableView; -public partial class VirtualizedVeryFastTableView : Panel, IKeyboardNavigationHandler +public partial class VirtualizedVeryFastTableView : RenderedPanel, ICustomKeyboardNavigation { protected static double ColumnSpacing = 10; protected static double RowHeight = 28; @@ -39,7 +43,7 @@ public partial class VirtualizedVeryFastTableView : Panel, IKeyboardNavigationHa private static bool GetResource(string key, T defaultVal, out T outT) { outT = defaultVal; - if (Application.Current!.Styles.TryGetResource(key, out var res) && res is T t) + if (Application.Current!.Styles.TryGetResource(key, SystemTheme.EffectiveThemeVariant, out var res) && res is T t) { outT = t; return true; @@ -139,16 +143,19 @@ protected override void OnAttachedToVisualTree(VisualTreeAttachmentEventArgs e) { InvalidateMeasure(); InvalidateArrange(); + InvalidateVisual(); }); if (ScrollViewer is { } sc) { sc.GetObservable(ScrollViewer.OffsetProperty).SubscribeAction(_ => { InvalidateArrange(); + InvalidateVisual(); }); sc.GetObservable(BoundsProperty).SubscribeAction(_ => { InvalidateArrange(); + InvalidateVisual(); }); } } @@ -213,13 +220,13 @@ protected override void OnTextInput(TextInputEventArgs e) e.Handled = true; } } - - protected override void OnPointerLeave(PointerEventArgs e) + + protected override void OnPointerExited(PointerEventArgs e) { - Cursor = Cursor.Default; + SetCurrentValue(CursorProperty, Cursor.Default); lastMouseLocation = new Point(-1, -1); InvalidateVisual(); - base.OnPointerLeave(e); + base.OnPointerExited(e); } protected override void OnPointerMoved(PointerEventArgs e) @@ -229,7 +236,7 @@ protected override void OnPointerMoved(PointerEventArgs e) var point = currentPoint.Position; lastMouseLocation = point; lastMouseButtonPressed = currentPoint.Properties.IsLeftButtonPressed; - Cursor = (currentlyResizedColumn.HasValue || IsPointHeader(point) && IsOverColumnSplitter(lastMouseLocation.X, out _)) ? new Cursor(StandardCursorType.SizeWestEast) : Cursor.Default; + SetCurrentValue(CursorProperty, (currentlyResizedColumn.HasValue || IsPointHeader(point) && IsOverColumnSplitter(lastMouseLocation.X, out _)) ? new Cursor(StandardCursorType.SizeWestEast) : Cursor.Default); if (Controller.UpdateCursor(currentPoint.Position, currentPoint.Properties.IsLeftButtonPressed)) InvalidateVisual(); @@ -267,7 +274,7 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) var index = GetRowIndexByY(e.GetPosition(this).Y); if (!IsRowIndexValid(index)) { - SelectedRowIndex = -1; + SetCurrentValue(SelectedRowIndexProperty, -1); MultiSelection.Clear(); } else @@ -308,16 +315,16 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) MultiSelection.Add(index); } } - SelectedRowIndex = index; + SetCurrentValue(SelectedRowIndexProperty, index); } } // cell { var index = GetColumnIndexByX(e.GetPosition(this).X); if (ItemsCount == 0 || !index.HasValue) - SelectedCellIndex = -1; + SetCurrentValue(SelectedCellIndexProperty, -1); else - SelectedCellIndex = Math.Clamp(index.Value, 0, ColumnsCount - 1); + SetCurrentValue(SelectedCellIndexProperty, Math.Clamp(index.Value, 0, ColumnsCount - 1)); } } else if (IsOverColumnSplitter(e.GetPosition(this).X, out var columnIndex)) @@ -337,9 +344,9 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) { var index = GetColumnIndexByX(e.GetPosition(this).X); if (ItemsCount == 0 || !index.HasValue) - SelectedCellIndex = -1; + SetCurrentValue(SelectedCellIndexProperty, -1); else - SelectedCellIndex = Math.Clamp(index.Value, 0, ColumnsCount - 1); + SetCurrentValue(SelectedCellIndexProperty, Math.Clamp(index.Value, 0, ColumnsCount - 1)); } } e.Handled = true; @@ -422,7 +429,6 @@ private void OpenEditor(string? customText = null) { var originalText = Controller.GetCellText(SelectedRowIndex, SelectedCellIndex); var initialText = customText ?? originalText ?? ""; - GetFonts(out var defaultFont, out var unicodeFallback); editor.Spawn(this, SelectedCellRect, initialText, customText == null, (text, action) => { if (originalText != text) @@ -432,19 +438,19 @@ private void OpenEditor(string? customText = null) switch (action) { case PhantomTextBox.ActionAfterSave.MoveUp: - Move(this, NavigationDirection.Up); + GetNext(this, NavigationDirection.Up); break; case PhantomTextBox.ActionAfterSave.MoveDown: - Move(this, NavigationDirection.Down); + GetNext(this, NavigationDirection.Down); break; case PhantomTextBox.ActionAfterSave.MoveNext: - Move(this, NavigationDirection.Next); + GetNext(this, NavigationDirection.Next); break; default: throw new ArgumentOutOfRangeException(nameof(action), action, null); } } - }, UseFallbackUnicodeFont(initialText) ? unicodeFallback : null); + }); } } @@ -502,7 +508,7 @@ private void UpdateKeyBindings() /*** * This is KeyBinding that forwards the gesture to the focused TextBox first */ - private class BetterKeyBinding : KeyBinding, ICommand + public class BetterKeyBinding : KeyBinding, ICommand { public static readonly StyledProperty CustomCommandProperty = AvaloniaProperty.Register(nameof (CustomCommand)); @@ -511,28 +517,52 @@ public ICommand CustomCommand get => GetValue(CustomCommandProperty); set => SetValue(CustomCommandProperty, value); } - + public BetterKeyBinding() { - Command = this; + SetCurrentValue(CommandProperty, this); + } + + public static TopLevel? GetTopLevel() + { + if (Application.Current?.ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop) + { + return desktop.MainWindow; + } + if (Application.Current?.ApplicationLifetime is ISingleViewApplicationLifetime viewApp) + { + var visualRoot = viewApp.MainView?.GetVisualRoot(); + return visualRoot as TopLevel; + } + return null; } public bool CanExecute(object? parameter) { - if (FocusManager.Instance!.Current is TextBox tb) - return true; + var focusManager = GetTopLevel()?.FocusManager; + var currentAsTextBox = focusManager?.GetFocusedElement() as TextBox; + var currentAsTextEditor = focusManager?.GetFocusedElement() as TextEditor; + var currentAsTextArea = focusManager?.GetFocusedElement() as TextArea; + if (currentAsTextBox != null || currentAsTextEditor != null || currentAsTextArea != null) + return true; return CustomCommand.CanExecute(parameter); } public void Execute(object? parameter) { - if (FocusManager.Instance!.Current is TextBox tb) + var focusManager = GetTopLevel()?.FocusManager; + var currentAsTextBox = focusManager?.GetFocusedElement() as TextBox; + var currentAsTextEditor = focusManager?.GetFocusedElement() as TextEditor; + var currentAsTextArea = focusManager?.GetFocusedElement() as TextArea; + if (currentAsTextBox != null || currentAsTextEditor != null || currentAsTextArea != null) { - var ev = Activator.CreateInstance(); - ev.Key = Gesture.Key; - ev.KeyModifiers = Gesture.KeyModifiers; - ev.RoutedEvent = InputElement.KeyDownEvent; - tb.RaiseEvent(ev); + var ev = new KeyEventArgs() + { + Key = Gesture.Key, + KeyModifiers = Gesture.KeyModifiers, + RoutedEvent = InputElement.KeyDownEvent + }; + ((Control?)currentAsTextBox ?? (Control?)currentAsTextEditor ?? currentAsTextArea)!.RaiseEvent(ev); if (!ev.Handled && CanExecute(parameter)) CustomCommand.Execute(parameter); } diff --git a/AvaloniaStyles/Controls/RecyclableViewList.cs b/AvaloniaStyles/Controls/RecyclableViewList.cs index e2bb8aa75..2a6b94d9d 100644 --- a/AvaloniaStyles/Controls/RecyclableViewList.cs +++ b/AvaloniaStyles/Controls/RecyclableViewList.cs @@ -11,7 +11,7 @@ public class RecyclableViewList { private readonly Panel owner; private readonly bool behind; - private List controls = new(); + private List controls = new(); private int counter = 0; private IDataTemplate? template; @@ -27,14 +27,14 @@ public void Reset(IDataTemplate? template) counter = 0; } - public bool TryGetNext(object context, out IControl control) + public bool TryGetNext(object context, out Control control) { control = null!; if (template == null) return false; if (counter >= controls.Count) { - control = template!.Build(null!); + control = template!.Build(null!)!; controls.Add(control); control.DataContext = context; if (behind) @@ -54,7 +54,7 @@ public bool TryGetNext(object context, out IControl control) controls[counter] = found; controls[i] = atCounter; - + ++counter; control = found; return true; @@ -64,6 +64,7 @@ public bool TryGetNext(object context, out IControl control) } control = controls[counter++]; + control.DataContext = null; control.DataContext = context; control.IsVisible = true; control.InvalidateMeasure(); @@ -71,7 +72,7 @@ public bool TryGetNext(object context, out IControl control) return true; } - public IControl GetNext(object context) + public Control GetNext(object context) { if (!TryGetNext(context, out var control)) throw new Exception("Template is null!"); @@ -89,4 +90,4 @@ public void Finish() template = null; } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/SettingItem.axaml b/AvaloniaStyles/Controls/SettingItem.axaml index cfaa558f5..0011c791c 100644 --- a/AvaloniaStyles/Controls/SettingItem.axaml +++ b/AvaloniaStyles/Controls/SettingItem.axaml @@ -5,13 +5,13 @@ - - - @@ -56,10 +56,10 @@ - - diff --git a/AvaloniaStyles/Controls/SettingItem.axaml.cs b/AvaloniaStyles/Controls/SettingItem.axaml.cs index 5fff2e1be..8d2041b9d 100644 --- a/AvaloniaStyles/Controls/SettingItem.axaml.cs +++ b/AvaloniaStyles/Controls/SettingItem.axaml.cs @@ -31,7 +31,7 @@ internal class SettingItemContentPresenter : ContentPresenter { protected override Size ArrangeOverride(Size finalSize) { - if (Child is TextBox || Child.HorizontalAlignment == HorizontalAlignment.Stretch) + if (Child is TextBox || Child != null && Child.HorizontalAlignment == HorizontalAlignment.Stretch) { if (HorizontalContentAlignment != HorizontalAlignment.Stretch) HorizontalContentAlignment = HorizontalAlignment.Stretch; diff --git a/AvaloniaStyles/Controls/StretchStackPanel.cs b/AvaloniaStyles/Controls/StretchStackPanel.cs index 3a4a311e5..c8c457756 100644 --- a/AvaloniaStyles/Controls/StretchStackPanel.cs +++ b/AvaloniaStyles/Controls/StretchStackPanel.cs @@ -14,7 +14,15 @@ public double Spacing get => GetValue(SpacingProperty); set => SetValue(SpacingProperty, value); } - + + public static readonly StyledProperty StretchProperty = AvaloniaProperty.Register(nameof(Stretch), true); + + public bool Stretch + { + get => GetValue(StretchProperty); + set => SetValue(StretchProperty, value); + } + protected override Size ArrangeOverride(Size finalSize) { Avalonia.Controls.Controls children = Children; @@ -22,7 +30,7 @@ protected override Size ArrangeOverride(Size finalSize) int index = 0; for (int count = children.Count; index < count; ++index) { - IControl control = children[index]; + Control control = children[index]; if (control == null) continue; @@ -33,11 +41,13 @@ protected override Size ArrangeOverride(Size finalSize) double singleWidth = (finalSize.Width - Spacing * Math.Max(0, visibleChildrenCount - 1)) / visibleChildrenCount; double x = 0; + double widthLeft = finalSize.Width; + bool stretch = Stretch; index = 0; for (int count = children.Count; index < count; ++index) { - IControl control = children[index]; + Control control = children[index]; if (control == null) continue; @@ -45,9 +55,11 @@ protected override Size ArrangeOverride(Size finalSize) if (!isVisible) continue; - - control.Arrange(new Rect(x, finalSize.Height / 2 - control.DesiredSize.Height/2, singleWidth, control.DesiredSize.Height)); - x += singleWidth + Spacing; + + var width = stretch ? singleWidth : Math.Clamp(control.DesiredSize.Width, 0, widthLeft); + control.Arrange(new Rect(x, finalSize.Height / 2 - control.DesiredSize.Height/2, width, control.DesiredSize.Height)); + x += width + Spacing; + widthLeft -= width + Spacing; } return finalSize; @@ -65,7 +77,7 @@ protected override Size MeasureOverride(Size availableSize) int index = 0; for (int count = children.Count; index < count; ++index) { - IControl control = children[index]; + Control control = children[index]; if (control != null) { bool isVisible = control.IsVisible; diff --git a/AvaloniaStyles/Controls/StretchWrapPanel.cs b/AvaloniaStyles/Controls/StretchWrapPanel.cs index ba0d3464c..4fefa37e6 100644 --- a/AvaloniaStyles/Controls/StretchWrapPanel.cs +++ b/AvaloniaStyles/Controls/StretchWrapPanel.cs @@ -35,9 +35,9 @@ protected override Size MeasureOverride(Size availableSize) for (var i = 0; i < visualCount; i++) { - IVisual visual = visualChildren[i]; + Visual visual = visualChildren[i]; - if (visual is not ILayoutable layoutable) + if (visual is not Layoutable layoutable) continue; layoutable.Measure(availableSize.WithWidth(Math.Max(availableSize.Width, 100))); @@ -87,9 +87,9 @@ protected override Size ArrangeOverride(Size finalSize) for (var i = 0; i < visualCount; i++) { - IVisual visual = visualChildren[i]; + Visual visual = visualChildren[i]; - if (visual is ILayoutable layoutable) + if (visual is Layoutable layoutable) { StyledElement? styledElement = layoutable as StyledElement; if (styledElement == null) diff --git a/AvaloniaStyles/Controls/ToolbarControl.axaml b/AvaloniaStyles/Controls/ToolbarControl.axaml index c21ebe691..f21e074ae 100644 --- a/AvaloniaStyles/Controls/ToolbarControl.axaml +++ b/AvaloniaStyles/Controls/ToolbarControl.axaml @@ -1,69 +1,98 @@ - - - - - - - + + diff --git a/AvaloniaStyles/Controls/ToolbarControl.axaml.cs b/AvaloniaStyles/Controls/ToolbarControl.axaml.cs deleted file mode 100644 index c891c15f7..000000000 --- a/AvaloniaStyles/Controls/ToolbarControl.axaml.cs +++ /dev/null @@ -1,78 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Collections.Specialized; -using System.Linq; -using Avalonia; -using Avalonia.Controls; -using Avalonia.Controls.Primitives; -using Avalonia.LogicalTree; -using Avalonia.Metadata; -using Avalonia.VisualTree; - -namespace AvaloniaStyles.Controls; - -public class ToolbarControl : TemplatedControl, IPanel -{ - public ToolbarControl() => this.Children.CollectionChanged += new NotifyCollectionChangedEventHandler(this.ChildrenChanged); - - public static readonly StyledProperty IsOverflowProperty = AvaloniaProperty.Register(nameof(IsOverflow)); - - public bool IsOverflow - { - get => GetValue(IsOverflowProperty); - set => SetValue(IsOverflowProperty, value); - } - - private Panel? childrenHost; - - protected override void OnApplyTemplate(TemplateAppliedEventArgs e) - { - base.OnApplyTemplate(e); - childrenHost = e.NameScope.Find("PART_ChildrenHost"); - foreach (var child in Children) - childrenHost.Children.Add(child); - } - - private void ChildrenChanged(object? sender, NotifyCollectionChangedEventArgs e) - { - switch (e.Action) - { - case NotifyCollectionChangedAction.Add: - List list = e.NewItems!.OfType().ToList(); - if (childrenHost != null) - { - foreach (Control control in list) - childrenHost.Children.Add(control); - } - break; - case NotifyCollectionChangedAction.Remove: - List oldList = e.OldItems!.OfType().ToList(); - if (childrenHost != null) - { - foreach (Control control in oldList) - childrenHost.Children.Remove(control); - } - break; - case NotifyCollectionChangedAction.Replace: - for (int index1 = 0; index1 < e.OldItems!.Count; ++index1) - { - int index2 = index1 + e.OldStartingIndex; - IControl? newItem = (IControl?) e.NewItems![index1]; - if (childrenHost != null) - childrenHost.Children[index2] = newItem; - } - break; - case NotifyCollectionChangedAction.Move: - if (childrenHost != null) - { - childrenHost.Children.MoveRange(e.OldStartingIndex, e.OldItems!.Count, e.NewStartingIndex); - } - break; - case NotifyCollectionChangedAction.Reset: - throw new NotSupportedException(); - } - } - - [Content] - public Avalonia.Controls.Controls Children { get; } = new Avalonia.Controls.Controls(); -} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/ToolbarItemsControl.cs b/AvaloniaStyles/Controls/ToolbarItemsControl.cs index 505784d2e..1c88091a1 100644 --- a/AvaloniaStyles/Controls/ToolbarItemsControl.cs +++ b/AvaloniaStyles/Controls/ToolbarItemsControl.cs @@ -1,8 +1,75 @@ +using System.Collections.Specialized; +using System.Linq; +using Avalonia; using Avalonia.Controls; +using Avalonia.Controls.Primitives; +using Avalonia.Metadata; namespace AvaloniaStyles.Controls; public class ToolbarItemsControl : ItemsControl { + public static readonly StyledProperty IsOverflowProperty = AvaloniaProperty.Register(nameof(IsOverflow)); + public bool IsOverflow + { + get => GetValue(IsOverflowProperty); + set => SetValue(IsOverflowProperty, value); + } +} + +public class ToolbarControl : TemplatedControl +{ + public static readonly StyledProperty IsOverflowProperty = AvaloniaProperty.Register(nameof(IsOverflow)); + + public bool IsOverflow + { + get => GetValue(IsOverflowProperty); + set => SetValue(IsOverflowProperty, value); + } + + private Panel? panel; + + protected override void OnApplyTemplate(TemplateAppliedEventArgs e) + { + base.OnApplyTemplate(e); + panel = e.NameScope.Get("PART_Panel"); + panel.Children.AddRange(Children); + } + + public ToolbarControl() => Children.CollectionChanged += ChildrenChanged; + + private void ChildrenChanged(object? sender, NotifyCollectionChangedEventArgs e) + { + if (panel == null) + return; + + if (e.Action == NotifyCollectionChangedAction.Add) + { + panel.Children.AddRange(e.NewItems!.OfType()); + } + else if (e.Action == NotifyCollectionChangedAction.Remove) + { + panel.Children.RemoveAll(e.OldItems!.OfType()); + } + else if (e.Action == NotifyCollectionChangedAction.Replace) + { + for (int index1 = 0; index1 < e.OldItems!.Count; ++index1) + { + int index2 = index1 + e.OldStartingIndex; + Control? newItem = (Control?) e.NewItems![index1]; + panel.Children[index2] = newItem!; + } + } + else + { + if (e.Action != NotifyCollectionChangedAction.Reset) + return; + panel.Children.Clear(); + panel.Children.AddRange(Children); + } + } + + [Content] + public Avalonia.Controls.Controls Children { get; } = new(); } \ No newline at end of file diff --git a/AvaloniaStyles/Controls/ToolbarPanel.cs b/AvaloniaStyles/Controls/ToolbarPanel.cs index e5e809796..6f4402e74 100644 --- a/AvaloniaStyles/Controls/ToolbarPanel.cs +++ b/AvaloniaStyles/Controls/ToolbarPanel.cs @@ -1,10 +1,12 @@ using System; using System.Collections.Generic; +using System.Reflection; using Avalonia; using Avalonia.Controls; using Avalonia.Controls.Presenters; using Avalonia.Layout; using Avalonia.VisualTree; +using WDE.Common; namespace AvaloniaStyles.Controls { @@ -15,7 +17,7 @@ public class ToolbarPanel : Panel public static readonly StyledProperty WrapOnOverflowProperty = AvaloniaProperty.Register(nameof(WrapOnOverflow)); public static readonly StyledProperty SpacingProperty = AvaloniaProperty.Register(nameof(Spacing), 4); - private List _overflowControls = new List(); + private List _overflowControls = new List(); public Panel? OutOfBoundsPanel { @@ -41,8 +43,17 @@ public double Spacing set => SetValue(SpacingProperty, value); } + delegate void InvalidateStylesType(StyledElement element, bool recurse); + private static InvalidateStylesType? invalidateStyle; + static ToolbarPanel() { + var invalidateStyleMethod = typeof(StyledElement).GetMethod("InvalidateStyles", BindingFlags.Instance | BindingFlags.NonPublic); + if (invalidateStyleMethod != null) + invalidateStyle = (InvalidateStylesType)Delegate.CreateDelegate(typeof(InvalidateStylesType), invalidateStyleMethod); + else + LOG.LogCritical("Failed to find InvalidateStyles method (probably due to Avalonia version update)!!!"); + OutOfBoundsPanelProperty.Changed.AddClassHandler((panel, e) => panel.OnChangedPanel(e)); AffectsMeasure(SpacingProperty); AffectsArrange(SpacingProperty); @@ -61,6 +72,8 @@ private void NewPanelOnAttachedToVisualTree(object? sender, VisualTreeAttachment foreach (var child in _overflowControls) { OutOfBoundsPanel!.Children.Insert(0, child); + // @todo ava11: is that necessary? + child.ApplyStyling(); } _overflowControls.Clear(); } @@ -99,6 +112,7 @@ protected override Size MeasureOverride(Size availableSize) int rows = 1; var children = Children; bool any = false; + for (int i = 0, count = children.Count; i < count; ++i) { var child = children[i]; @@ -245,8 +259,8 @@ protected override Size ArrangeOverride(Size finalSize) rcChild = rcChild.WithHeight(Math.Max(finalSize.Height, child.DesiredSize.Height)); // we want to stretch only content presenters, not buttons // if it doesn't work for future toolbars, it can be changed - if (child.HorizontalAlignment == HorizontalAlignment.Stretch && child is ContentPresenter cp && - cp.Child is not Button) + if (child.HorizontalAlignment == HorizontalAlignment.Stretch && + (child is ContentPresenter { Child: not Button } || (child is ContentControl))) { previousChildSize = child.DesiredSize.Width + Math.Max(0, leftSpace); rcChild = rcChild.WithWidth(previousChildSize); @@ -274,7 +288,7 @@ protected override Size ArrangeOverride(Size finalSize) { var c = children[^1]; Children.RemoveAt(Children.Count - 1); - if (((IVisual)panel).IsAttachedToVisualTree) + if (panel.IsAttachedToVisualTree()) { panel.Children.Insert(0, c); ((ISetInheritanceParent)c).SetParent(this); @@ -282,7 +296,7 @@ protected override Size ArrangeOverride(Size finalSize) else _overflowControls.Add(c); } - IsOverflow = true; + SetCurrentValue(IsOverflowProperty, true); return finalSize; } } @@ -293,7 +307,7 @@ protected override Size ArrangeOverride(Size finalSize) { if (leftSpace > 0) { - IControl? child = null; + Control? child = null; if (_overflowControls.Count > 0) { child = _overflowControls[^1]; @@ -312,10 +326,17 @@ protected override Size ArrangeOverride(Size finalSize) else OutOfBoundsPanel.Children.RemoveAt(0); Children.Add(child); + // it really sucks I have to call InvalidateStyles via reflection, but without this + // buttons style bug in Solution Explorer in the following case + // shrink the toolbar to overflow buttons + // open the flyout + // expand the toolbar to show all buttons - now the buttons are not styled + invalidateStyle?.Invoke(child, true); + child.ApplyStyling(); } } } - IsOverflow = OutOfBoundsPanel.Children.Count > 0 || _overflowControls.Count > 0; + SetCurrentValue(IsOverflowProperty, OutOfBoundsPanel.Children.Count > 0 || _overflowControls.Count > 0); } return finalSize; @@ -370,7 +391,7 @@ private Size ArrangeWithWrapOverride(Size finalSize) } internal virtual void ArrangeChild( - IControl child, + Control child, Rect rect, Size panelSize) { @@ -381,4 +402,4 @@ internal virtual void ArrangeChild( public class ToolbarSpacer : Control { } -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/ToolsTabControl.cs b/AvaloniaStyles/Controls/ToolsTabControl.cs index c5efa542e..8b1378917 100644 --- a/AvaloniaStyles/Controls/ToolsTabControl.cs +++ b/AvaloniaStyles/Controls/ToolsTabControl.cs @@ -1,128 +1 @@ -using System.Collections; -using System.Collections.Generic; -using System.Collections.Specialized; -using Avalonia; -using Avalonia.Controls; -using Avalonia.Controls.Generators; -using Avalonia.Controls.Presenters; -using Avalonia.Controls.Templates; -using Avalonia.Styling; -namespace AvaloniaStyles.Controls -{ - public class ToolsTabControl : TabControl, IStyleable - { - public static readonly AvaloniaProperty AttachedViewProperty = - AvaloniaProperty.RegisterAttached("AttachedView"); - - public static IControl? GetAttachedView(TabItem control) => (IControl?) control.GetValue(AttachedViewProperty); - - public static void SetAttachedView(TabItem control, IControl? value) => - control.SetValue(AttachedViewProperty, value); - - - public static readonly AvaloniaProperty ToolTitleProperty = - AvaloniaProperty.RegisterAttached("ToolTitle"); - public static string? GetToolTitle(TabItem control) => (string?) control.GetValue(ToolTitleProperty); - public static void SetToolTitle(TabItem control, string value) => - control.SetValue(ToolTitleProperty, value); - - - public static readonly StyledProperty ActiveViewProperty = - AvaloniaProperty.Register(nameof(ActiveView)); - - public IControl? ActiveView - { - get => GetValue(ActiveViewProperty); - internal set => SetValue(ActiveViewProperty, value); - } - - internal IContentPresenter? ContentPart { get; private set; } - private static readonly FuncTemplate DefaultPanel = - new FuncTemplate(() => new StackPanel(){Name = "ToolsList"}); - - static ToolsTabControl() - { - ItemsPanelProperty.OverrideDefaultValue(DefaultPanel); - - SelectedContentProperty.Changed.AddClassHandler((viewModelItemsControl, e) => - { - if (viewModelItemsControl.SelectedItem != null && - viewModelItemsControl.itemsToViews.TryGetValue(viewModelItemsControl.SelectedItem, out var view)) - viewModelItemsControl.ActiveView = view; - else - viewModelItemsControl.ActiveView = null; - if (viewModelItemsControl.ContentPart != null) - viewModelItemsControl.ContentPart.DataContext = viewModelItemsControl.SelectedItem; - }); - } - - protected override bool RegisterContentPresenter(IContentPresenter presenter) - { - base.RegisterContentPresenter(presenter); - if (presenter.Name == "PART_SelectedContentHost") - { - ContentPart = presenter; - return true; - } - - return false; - } - - protected override void OnContainersMaterialized(ItemContainerEventArgs e) - { - base.OnContainersMaterialized(e); - - foreach (var container in e.Containers) - { - if (container.Item != null && itemsToViews.TryGetValue(container.Item, out var vieww)) - SetAttachedView((TabItem)container.ContainerControl, vieww as IControl); - } - } - - private Dictionary itemsToViews = new(); - - protected override void ItemsChanged(AvaloniaPropertyChangedEventArgs e) - { - if (e.NewValue != null) - GenerateContent((IList)e.NewValue); - base.ItemsChanged(e); - } - - protected override void ItemsCollectionChanged(object sender, NotifyCollectionChangedEventArgs e) - { - switch (e.Action) - { - case NotifyCollectionChangedAction.Add: - GenerateContent(e.NewItems!); - break; - - case NotifyCollectionChangedAction.Remove: - FreeContent(e.OldItems!); - break; - } - - base.ItemsCollectionChanged(sender, e); - } - - private void FreeContent(IList eOldItems) - { - foreach (var o in eOldItems) - itemsToViews.Remove(o); - } - - private void GenerateContent(IList eNewItems) - { - foreach (var o in eNewItems) - { - IControl view = GenerateView(o); - itemsToViews[o] = view; - } - } - - protected virtual IControl GenerateView(object viewModel) - { - return new TextBlock() {Text = viewModel.ToString()}; - } - } -} \ No newline at end of file diff --git a/AvaloniaStyles/Controls/TwoColumnsPanel.cs b/AvaloniaStyles/Controls/TwoColumnsPanel.cs index f8ff64d86..daa8de98d 100644 --- a/AvaloniaStyles/Controls/TwoColumnsPanel.cs +++ b/AvaloniaStyles/Controls/TwoColumnsPanel.cs @@ -44,12 +44,12 @@ static TwoColumnsPanel() VerticalAlignmentProperty.OverrideDefaultValue(VerticalAlignment.Top); } - private IEnumerable<(IControl left, IControl? right)> GetChildren() + private IEnumerable<(Control left, Control? right)> GetChildren() { for (var index = 0; index < Children.Count; index++) { - IControl left; - IControl? right = null; + Control left; + Control? right = null; left = Children[index]; if (left.IsSet(ColumnSpanProperty)) diff --git a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridCheckBox.cs b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridCheckBox.cs index 41d7ea84f..d950e1a79 100644 --- a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridCheckBox.cs +++ b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridCheckBox.cs @@ -8,7 +8,7 @@ namespace AvaloniaStyles.Controls; -public class VirtualizedGridCheckBox : CheckBox, IStyleable +public class VirtualizedGridCheckBox : CheckBox { private IMultiIndexContainer? checkedIndices; public static readonly DirectProperty CheckedIndicesProperty = @@ -30,7 +30,7 @@ public int Index set => SetAndRaise(IndexProperty, ref index, value); } - Type IStyleable.StyleKey => typeof(CheckBox); + protected override Type StyleKeyOverride => typeof(CheckBox); static VirtualizedGridCheckBox() { diff --git a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridView.cs b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridView.cs index e7cb909a6..5e3f66ffe 100644 --- a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridView.cs +++ b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridView.cs @@ -1,5 +1,6 @@ using System; using System.Collections.Generic; +using System.Runtime.InteropServices; using System.Linq; using Avalonia; using Avalonia.Collections; @@ -9,6 +10,7 @@ using Avalonia.Input; using Avalonia.Input.Platform; using Avalonia.Interactivity; +using Avalonia.Utilities; using Avalonia.VisualTree; using DynamicData; using WDE.Common.Collections; @@ -46,12 +48,18 @@ public class VirtualizedGridView : TemplatedControl public static readonly RoutedEvent ColumnWidthChangedEvent = RoutedEvent.Register(nameof(ColumnWidthChanged), RoutingStrategies.Bubble); + static VirtualizedGridView() + { + FocusableProperty.OverrideDefaultValue(true); + } + public event EventHandler ColumnWidthChanged { add => AddHandler(ColumnWidthChangedEvent, value); remove => RemoveHandler(ColumnWidthChangedEvent, value); } - + +#pragma warning disable AVP1030 public int? FocusedIndex { get @@ -63,6 +71,8 @@ public int? FocusedIndex } set => SetValue(FocusedIndexProperty, value); } +#pragma warning restore AVP1030 + public double ItemHeight { get => itemHeight; @@ -107,6 +117,8 @@ public IEnumerable Columns private VirtualizedGridViewItemPresenter? children; private Grid? header; + private ScrollViewer? headerScroll; + private ScrollViewer? contentScroll; public VirtualizedGridViewItemPresenter? ItemPresenter => children; @@ -119,7 +131,7 @@ protected override void OnKeyDown(KeyEventArgs e) if (e.Key is Key.Up or Key.Down) { var rangeModifier = e.KeyModifiers.HasFlagFast(KeyModifiers.Shift); - var toggleModifier = e.KeyModifiers.HasFlagFast(AvaloniaLocator.Current.GetRequiredService().CommandModifiers); + var toggleModifier = e.KeyModifiers.HasFlagFast(RuntimeInformation.IsOSPlatform(OSPlatform.OSX) ? KeyModifiers.Meta : KeyModifiers.Control); int index; if (e.Key == Key.Up) @@ -142,7 +154,7 @@ protected override void OnKeyDown(KeyEventArgs e) protected ListBoxItem? GetContainerFromEventSource(object? eventSource) { - for (var current = eventSource as IVisual; current != null; current = current.GetVisualParent()) + for (var current = eventSource as Visual; current != null; current = current.GetVisualParent()) { if (current is ListBoxItem lbi) { @@ -168,11 +180,21 @@ public IEnumerable GetColumnsWidth() protected override void OnApplyTemplate(TemplateAppliedEventArgs e) { - header = e.NameScope.Find("PART_header"); - children = e.NameScope.Find("PART_Children"); + if (headerScroll is not null) + headerScroll.ScrollChanged -= OnHeaderScroll; + if (contentScroll is not null) + contentScroll.ScrollChanged -= OnContentScroll; + + header = e.NameScope.Get("PART_header"); + children = e.NameScope.Get("PART_Children"); + headerScroll = e.NameScope.Get("PART_HeaderScroll"); + contentScroll = e.NameScope.Get("PART_ContentScroll"); header.ColumnDefinitions.Clear(); header.Children.Clear(); + + headerScroll.ScrollChanged += OnHeaderScroll; + contentScroll.ScrollChanged += OnContentScroll; VirtualizedGridViewItemPresenter.SetupGridColumns(columns, header, true, useCheckBoxes); @@ -207,6 +229,18 @@ void AddHeaderCell(string name) AddHeaderCell(column.Name); } + private void OnContentScroll(object? sender, ScrollChangedEventArgs e) + { + if (contentScroll is not null && headerScroll is not null && !MathUtilities.IsZero(e.OffsetDelta.X)) + headerScroll.Offset = headerScroll.Offset.WithX(contentScroll.Offset.X); + } + + private void OnHeaderScroll(object? sender, ScrollChangedEventArgs e) + { + if (contentScroll is not null && headerScroll is not null && !MathUtilities.IsZero(e.OffsetDelta.X)) + contentScroll.Offset = contentScroll.Offset.WithX(headerScroll.Offset.X); + } + private void ColumnsSizeChanged() { RaiseEvent(new RoutedEventArgs() @@ -272,7 +306,7 @@ private void UpdateSelectionFromIndex(bool rangeModifier, bool toggleModifier, i for (int i = index; i <= oldFocusedIndex; ++i) selection.Add(i); } - FocusedIndex = index; + SetCurrentValue(FocusedIndexProperty, index); } else { @@ -280,7 +314,7 @@ private void UpdateSelectionFromIndex(bool rangeModifier, bool toggleModifier, i selection.Remove(index); else selection.Add(index); - FocusedIndex = index; + SetCurrentValue(FocusedIndexProperty, index); } } @@ -299,7 +333,7 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) e.Source, true, e.KeyModifiers.HasFlagFast(KeyModifiers.Shift), - e.KeyModifiers.HasFlagFast(AvaloniaLocator.Current.GetRequiredService().CommandModifiers), + e.KeyModifiers.HasFlagFast(RuntimeInformation.IsOSPlatform(OSPlatform.OSX) ? KeyModifiers.Meta : KeyModifiers.Control), point.Properties.IsRightButtonPressed); } } @@ -307,4 +341,4 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) private ScrollViewer? ScrollViewer => this.FindDescendantOfType(); -} \ No newline at end of file +} diff --git a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewItemPresenter.cs b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewItemPresenter.cs index 54ee845b6..51d5e50ba 100644 --- a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewItemPresenter.cs +++ b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewItemPresenter.cs @@ -8,6 +8,7 @@ using Avalonia.Controls.Primitives; using Avalonia.Controls.Templates; using Avalonia.Data; +using Avalonia.ReactiveUI; using Avalonia.Threading; using Avalonia.VisualTree; using WDE.Common.Collections; @@ -64,10 +65,10 @@ public VirtualizedGridViewItemPresenter() foreach (var column in Columns) { - IControl control; + Control control; if (column.DataTemplate is { } dt) { - control = dt.Build(column); + control = dt.Build(column) ?? new TextBlock(){Text = "Error: data template for column returned null control"}; } else if (column.Checkable) { @@ -181,7 +182,8 @@ public IIndexedCollection? Items get => items; set => SetAndRaise(ItemsProperty, ref items, value); } - + + #pragma warning disable AVP1030 public int? FocusedIndex { get @@ -193,6 +195,7 @@ public int? FocusedIndex } set => SetValue(FocusedIndexProperty, value); } + #pragma warning restore AVP1030 public IMultiIndexContainer Selection { diff --git a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewStyle.axaml b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewStyle.axaml index 4ce17188f..86f87e77b 100644 --- a/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewStyle.axaml +++ b/AvaloniaStyles/Controls/VirtualizedGridViewControl/VirtualizedGridViewStyle.axaml @@ -9,7 +9,7 @@ - - - - - - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/BigSur/CheckBox.xaml b/AvaloniaStyles/Styles/BigSur/CheckBox.xaml index 0f5b4ceb8..aa19ac30a 100644 --- a/AvaloniaStyles/Styles/BigSur/CheckBox.xaml +++ b/AvaloniaStyles/Styles/BigSur/CheckBox.xaml @@ -36,7 +36,7 @@ diff --git a/AvaloniaStyles/Styles/BigSur/ContextMenu.xaml b/AvaloniaStyles/Styles/BigSur/ContextMenu.xaml index 232393cbb..d8738993b 100644 --- a/AvaloniaStyles/Styles/BigSur/ContextMenu.xaml +++ b/AvaloniaStyles/Styles/BigSur/ContextMenu.xaml @@ -3,8 +3,8 @@ - - + + diff --git a/AvaloniaStyles/Styles/BigSur/GroupingListBox.axaml b/AvaloniaStyles/Styles/BigSur/GroupingListBox.axaml index d3b3802e3..4844447e3 100644 --- a/AvaloniaStyles/Styles/BigSur/GroupingListBox.axaml +++ b/AvaloniaStyles/Styles/BigSur/GroupingListBox.axaml @@ -30,7 +30,7 @@ - - - - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Catalina/CheckBox.xaml b/AvaloniaStyles/Styles/Catalina/CheckBox.xaml index 6e092b9a8..ad3a799c6 100644 --- a/AvaloniaStyles/Styles/Catalina/CheckBox.xaml +++ b/AvaloniaStyles/Styles/Catalina/CheckBox.xaml @@ -36,7 +36,7 @@ diff --git a/AvaloniaStyles/Styles/Catalina/ContextMenu.xaml b/AvaloniaStyles/Styles/Catalina/ContextMenu.xaml index a9b98a24d..69b6ad9ff 100644 --- a/AvaloniaStyles/Styles/Catalina/ContextMenu.xaml +++ b/AvaloniaStyles/Styles/Catalina/ContextMenu.xaml @@ -3,8 +3,8 @@ - - + + diff --git a/AvaloniaStyles/Styles/Catalina/GroupingListBox.axaml b/AvaloniaStyles/Styles/Catalina/GroupingListBox.axaml index d3b3802e3..4844447e3 100644 --- a/AvaloniaStyles/Styles/Catalina/GroupingListBox.axaml +++ b/AvaloniaStyles/Styles/Catalina/GroupingListBox.axaml @@ -30,7 +30,7 @@ + + - + \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/ColorsDark.xaml b/AvaloniaStyles/Styles/Windows10/ColorsDark.xaml deleted file mode 100644 index c9d1f2ab9..000000000 --- a/AvaloniaStyles/Styles/Windows10/ColorsDark.xaml +++ /dev/null @@ -1,106 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 0 - 0 - 7,5,7,7 - 8,5,8,6 - 0.5 - 0.5 - - - - - - - - - - - - - - - - - - - - - #007ACC - #1C97EA - #52B0EF - #F0F0F0 - #007ACC - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/AvaloniaStyles/Styles/Windows10/ColorsLight.xaml b/AvaloniaStyles/Styles/Windows10/ColorsLight.axaml similarity index 61% rename from AvaloniaStyles/Styles/Windows10/ColorsLight.xaml rename to AvaloniaStyles/Styles/Windows10/ColorsLight.axaml index b391beb26..a20b7f38b 100644 --- a/AvaloniaStyles/Styles/Windows10/ColorsLight.xaml +++ b/AvaloniaStyles/Styles/Windows10/ColorsLight.axaml @@ -1,6 +1,9 @@ + + + @@ -11,9 +14,9 @@ - + - + @@ -129,11 +132,49 @@ - + - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -42,18 +42,18 @@ diff --git a/AvaloniaStyles/Styles/Windows10/Dock.axaml b/AvaloniaStyles/Styles/Windows10/Dock.axaml new file mode 100644 index 000000000..fa805dbb3 --- /dev/null +++ b/AvaloniaStyles/Styles/Windows10/Dock.axaml @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + diff --git a/AvaloniaStyles/Styles/Windows10/DockLight.axaml b/AvaloniaStyles/Styles/Windows10/DockLight.axaml index 773195736..05feb0c8b 100644 --- a/AvaloniaStyles/Styles/Windows10/DockLight.axaml +++ b/AvaloniaStyles/Styles/Windows10/DockLight.axaml @@ -14,26 +14,26 @@ - - - - - - @@ -43,7 +43,7 @@ - diff --git a/AvaloniaStyles/Styles/Windows10/ListBox.xaml b/AvaloniaStyles/Styles/Windows10/ListBox.axaml similarity index 100% rename from AvaloniaStyles/Styles/Windows10/ListBox.xaml rename to AvaloniaStyles/Styles/Windows10/ListBox.axaml diff --git a/AvaloniaStyles/Styles/Windows10/NativeMenuBar.xaml b/AvaloniaStyles/Styles/Windows10/NativeMenuBar.xaml deleted file mode 100644 index eddad65a2..000000000 --- a/AvaloniaStyles/Styles/Windows10/NativeMenuBar.xaml +++ /dev/null @@ -1,26 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/SideBar.xaml b/AvaloniaStyles/Styles/Windows10/SideBar.axaml similarity index 100% rename from AvaloniaStyles/Styles/Windows10/SideBar.xaml rename to AvaloniaStyles/Styles/Windows10/SideBar.axaml diff --git a/AvaloniaStyles/Styles/Windows10/StatusBar.xaml b/AvaloniaStyles/Styles/Windows10/StatusBar.axaml similarity index 88% rename from AvaloniaStyles/Styles/Windows10/StatusBar.xaml rename to AvaloniaStyles/Styles/Windows10/StatusBar.axaml index 4fa24de30..4e7238075 100644 --- a/AvaloniaStyles/Styles/Windows10/StatusBar.xaml +++ b/AvaloniaStyles/Styles/Windows10/StatusBar.axaml @@ -20,7 +20,7 @@ \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/Style.xaml b/AvaloniaStyles/Styles/Windows10/Style.axaml similarity index 70% rename from AvaloniaStyles/Styles/Windows10/Style.xaml rename to AvaloniaStyles/Styles/Windows10/Style.axaml index 84d6e4bb2..c485a6e30 100644 --- a/AvaloniaStyles/Styles/Windows10/Style.xaml +++ b/AvaloniaStyles/Styles/Windows10/Style.axaml @@ -3,10 +3,16 @@ xmlns:system="clr-namespace:System;assembly=System.Runtime"> @@ -15,17 +21,15 @@ - - - + - + @@ -36,10 +40,11 @@ - - - + + + + \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/TabControl.axaml b/AvaloniaStyles/Styles/Windows10/TabControl.axaml index aee305e0f..b7d50664e 100644 --- a/AvaloniaStyles/Styles/Windows10/TabControl.axaml +++ b/AvaloniaStyles/Styles/Windows10/TabControl.axaml @@ -23,9 +23,7 @@ @@ -79,9 +77,9 @@ Content="{TemplateBinding Header}" HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}" VerticalAlignment="{TemplateBinding VerticalContentAlignment}" - TextBlock.FontFamily="{TemplateBinding FontFamily}" - TextBlock.FontSize="{TemplateBinding FontSize}" - TextBlock.FontWeight="{TemplateBinding FontWeight}" /> + TextElement.FontFamily="{TemplateBinding FontFamily}" + TextElement.FontSize="{TemplateBinding FontSize}" + TextElement.FontWeight="{TemplateBinding FontWeight}" /> @@ -102,7 +100,7 @@ @@ -139,7 +137,7 @@ @@ -152,4 +150,9 @@ - \ No newline at end of file + + + + diff --git a/AvaloniaStyles/Styles/Windows10/TabStrip.xaml b/AvaloniaStyles/Styles/Windows10/TabStrip.axaml similarity index 96% rename from AvaloniaStyles/Styles/Windows10/TabStrip.xaml rename to AvaloniaStyles/Styles/Windows10/TabStrip.axaml index a6d072cae..b4efc30ab 100644 --- a/AvaloniaStyles/Styles/Windows10/TabStrip.xaml +++ b/AvaloniaStyles/Styles/Windows10/TabStrip.axaml @@ -10,9 +10,7 @@ + ItemsPanel="{TemplateBinding ItemsPanel}" /> diff --git a/AvaloniaStyles/Styles/Windows10/ToolbarPanel.axaml b/AvaloniaStyles/Styles/Windows10/ToolbarPanel.axaml index 775c7c183..ab41fd19b 100644 --- a/AvaloniaStyles/Styles/Windows10/ToolbarPanel.axaml +++ b/AvaloniaStyles/Styles/Windows10/ToolbarPanel.axaml @@ -34,6 +34,10 @@ + @@ -41,7 +45,8 @@ - + + + @@ -69,6 +78,39 @@ + + + + + + + + + + + + @@ -96,29 +142,66 @@ + + + + + + + + + + - - - - - + - @@ -131,6 +214,7 @@ - - - - - - - - - - - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/TreeViewItem.axaml b/AvaloniaStyles/Styles/Windows10/TreeViewItem.axaml index 6ca4e4605..2b546abaf 100644 --- a/AvaloniaStyles/Styles/Windows10/TreeViewItem.axaml +++ b/AvaloniaStyles/Styles/Windows10/TreeViewItem.axaml @@ -6,16 +6,16 @@ - + - + - + 28 16 @@ -25,44 +25,4 @@ Left="True" x:Key="TreeViewItemLeftMarginConverter" /> - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/Window.axaml b/AvaloniaStyles/Styles/Windows10/Window.axaml new file mode 100644 index 000000000..ef6058247 --- /dev/null +++ b/AvaloniaStyles/Styles/Windows10/Window.axaml @@ -0,0 +1,159 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/AvaloniaStyles/Styles/Windows10/Window.xaml b/AvaloniaStyles/Styles/Windows10/Window.xaml deleted file mode 100644 index 81d4dbd7c..000000000 --- a/AvaloniaStyles/Styles/Windows10/Window.xaml +++ /dev/null @@ -1,164 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10/WindowMessageBox.axaml b/AvaloniaStyles/Styles/Windows10/WindowMessageBox.axaml new file mode 100644 index 000000000..054773d9a --- /dev/null +++ b/AvaloniaStyles/Styles/Windows10/WindowMessageBox.axaml @@ -0,0 +1,117 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows10Dark.xaml b/AvaloniaStyles/Styles/Windows10Dark.axaml similarity index 74% rename from AvaloniaStyles/Styles/Windows10Dark.xaml rename to AvaloniaStyles/Styles/Windows10Dark.axaml index dde0312ff..e414db22d 100644 --- a/AvaloniaStyles/Styles/Windows10Dark.xaml +++ b/AvaloniaStyles/Styles/Windows10Dark.axaml @@ -1,6 +1,6 @@ - - - + + + diff --git a/AvaloniaStyles/Styles/Windows10Light.xaml b/AvaloniaStyles/Styles/Windows10Light.axaml similarity index 72% rename from AvaloniaStyles/Styles/Windows10Light.xaml rename to AvaloniaStyles/Styles/Windows10Light.axaml index 6297403cf..3169bf442 100644 --- a/AvaloniaStyles/Styles/Windows10Light.xaml +++ b/AvaloniaStyles/Styles/Windows10Light.axaml @@ -1,7 +1,8 @@ - - - + + + diff --git a/AvaloniaStyles/Styles/Windows10Light.cs b/AvaloniaStyles/Styles/Windows10Light.cs new file mode 100644 index 000000000..9ed579496 --- /dev/null +++ b/AvaloniaStyles/Styles/Windows10Light.cs @@ -0,0 +1,12 @@ +using System; +using Avalonia.Markup.Xaml; + +namespace AvaloniaStyles.Styles; + +public class Windows10Light : Avalonia.Styling.Styles +{ + public Windows10Light(IServiceProvider? sp = null) + { + AvaloniaXamlLoader.Load(sp, this); + } +} \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows11/Button.axaml b/AvaloniaStyles/Styles/Windows11/Button.axaml index b9bffc191..b50ace178 100644 --- a/AvaloniaStyles/Styles/Windows11/Button.axaml +++ b/AvaloniaStyles/Styles/Windows11/Button.axaml @@ -2,27 +2,29 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:controls="clr-namespace:AvaloniaStyles.Controls" xmlns:utils="clr-namespace:AvaloniaStyles.Utils"> - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows11/ColorsDark.xaml b/AvaloniaStyles/Styles/Windows11/ColorsDark.axaml similarity index 62% rename from AvaloniaStyles/Styles/Windows11/ColorsDark.xaml rename to AvaloniaStyles/Styles/Windows11/ColorsDark.axaml index 97f72cb83..8765a59e8 100644 --- a/AvaloniaStyles/Styles/Windows11/ColorsDark.xaml +++ b/AvaloniaStyles/Styles/Windows11/ColorsDark.axaml @@ -1,16 +1,16 @@ - + - - - + + + - + @@ -39,12 +39,12 @@ - - - - - - + + + + + + 0,0,0,1 @@ -71,17 +71,17 @@ #19FFFFFF #33FFFFFF - - - - + + + + - + diff --git a/AvaloniaStyles/Styles/Windows11/ColorsLight.xaml b/AvaloniaStyles/Styles/Windows11/ColorsLight.axaml similarity index 61% rename from AvaloniaStyles/Styles/Windows11/ColorsLight.xaml rename to AvaloniaStyles/Styles/Windows11/ColorsLight.axaml index 57c709f60..e7f918c7e 100644 --- a/AvaloniaStyles/Styles/Windows11/ColorsLight.xaml +++ b/AvaloniaStyles/Styles/Windows11/ColorsLight.axaml @@ -3,19 +3,19 @@ xmlns:sys="clr-namespace:System;assembly=netstandard" xmlns:styles="clr-namespace:AvaloniaStyles.Styles" xmlns:utils="clr-namespace:AvaloniaStyles.Utils"> - + - - - + + + - + - + @@ -37,8 +37,8 @@ #086FBF #74b9ff--> - - + + @@ -47,12 +47,12 @@ - - - - - - + + + + + + 0,0,0,1 @@ -68,11 +68,11 @@ #19000000 #33000000 - - - - + + + + @@ -90,6 +90,6 @@ - + diff --git a/AvaloniaStyles/Styles/Windows11/DockLight.axaml b/AvaloniaStyles/Styles/Windows11/DockLight.axaml index e23b05094..964f58d47 100644 --- a/AvaloniaStyles/Styles/Windows11/DockLight.axaml +++ b/AvaloniaStyles/Styles/Windows11/DockLight.axaml @@ -2,9 +2,9 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:idc="clr-namespace:Dock.Avalonia.Controls;assembly=Dock.Avalonia"> - - - + + + - - - - - - @@ -48,11 +48,11 @@ - - diff --git a/AvaloniaStyles/Styles/Windows11/NativeMenuBar.xaml b/AvaloniaStyles/Styles/Windows11/NativeMenuBar.xaml deleted file mode 100644 index c72433dca..000000000 --- a/AvaloniaStyles/Styles/Windows11/NativeMenuBar.xaml +++ /dev/null @@ -1,48 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows11/Style.xaml b/AvaloniaStyles/Styles/Windows11/Style.axaml similarity index 71% rename from AvaloniaStyles/Styles/Windows11/Style.xaml rename to AvaloniaStyles/Styles/Windows11/Style.axaml index 6cf7c5f48..a60b95a79 100644 --- a/AvaloniaStyles/Styles/Windows11/Style.xaml +++ b/AvaloniaStyles/Styles/Windows11/Style.axaml @@ -3,10 +3,17 @@ xmlns:system="clr-namespace:System;assembly=System.Runtime"> @@ -15,18 +22,18 @@ - - - + + + + - - + - + @@ -37,15 +44,13 @@ - - - - + + \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows11/TabControl.axaml b/AvaloniaStyles/Styles/Windows11/TabControl.axaml index 072e25565..6c30fb5f6 100644 --- a/AvaloniaStyles/Styles/Windows11/TabControl.axaml +++ b/AvaloniaStyles/Styles/Windows11/TabControl.axaml @@ -3,7 +3,7 @@ diff --git a/AvaloniaStyles/Styles/Windows11/TextBox.axaml b/AvaloniaStyles/Styles/Windows11/TextBox.axaml index f30ad9cd8..b4a0fcb78 100644 --- a/AvaloniaStyles/Styles/Windows11/TextBox.axaml +++ b/AvaloniaStyles/Styles/Windows11/TextBox.axaml @@ -3,136 +3,159 @@ xmlns:controls="clr-namespace:AvaloniaStyles.Controls" xmlns:utils="clr-namespace:AvaloniaStyles.Utils" xmlns:avaloniaEdit="https://github.com/avaloniaui/avaloniaedit"> - + - + - - - + + + - + - + - + - - - + - + + + + + + + + + + + + + + - - - - + + + + + + + \ No newline at end of file diff --git a/AvaloniaStyles/Styles/Windows11/Window.axaml b/AvaloniaStyles/Styles/Windows11/Window.axaml index 31f8d158a..fd16419c6 100644 --- a/AvaloniaStyles/Styles/Windows11/Window.axaml +++ b/AvaloniaStyles/Styles/Windows11/Window.axaml @@ -9,9 +9,9 @@ - - - + + + diff --git a/AvaloniaStyles/Styles/Windows11Dark.xaml b/AvaloniaStyles/Styles/Windows11Dark.axaml similarity index 74% rename from AvaloniaStyles/Styles/Windows11Dark.xaml rename to AvaloniaStyles/Styles/Windows11Dark.axaml index 36e8a8ef9..d4cc3fd7e 100644 --- a/AvaloniaStyles/Styles/Windows11Dark.xaml +++ b/AvaloniaStyles/Styles/Windows11Dark.axaml @@ -1,6 +1,6 @@ - - - + + + diff --git a/AvaloniaStyles/Styles/Windows11Light.xaml b/AvaloniaStyles/Styles/Windows11Light.axaml similarity index 79% rename from AvaloniaStyles/Styles/Windows11Light.xaml rename to AvaloniaStyles/Styles/Windows11Light.axaml index b2bf08d80..ffb0cfef9 100644 --- a/AvaloniaStyles/Styles/Windows11Light.xaml +++ b/AvaloniaStyles/Styles/Windows11Light.axaml @@ -1,7 +1,7 @@ - - - + + + diff --git a/AvaloniaStyles/SystemTheme.cs b/AvaloniaStyles/SystemTheme.cs index a6f0081e8..536e68274 100644 --- a/AvaloniaStyles/SystemTheme.cs +++ b/AvaloniaStyles/SystemTheme.cs @@ -1,7 +1,9 @@ using System; using System.Runtime.InteropServices; +using Avalonia; using Avalonia.Markup.Xaml.Styling; using Avalonia.Metadata; +using Avalonia.Styling; [assembly: XmlnsDefinition("https://github.com/avaloniaui", "AvaloniaStyles.Controls")] @@ -73,6 +75,10 @@ public SystemThemeOptions Mode // || EffectiveTheme == SystemThemeOptions.MacOsCatalinaDark || // EffectiveTheme == SystemThemeOptions.MacOsBigSurDark; + Application.Current!.RequestedThemeVariant = EffectiveThemeIsDark + ? ThemeVariant.Dark + : ThemeVariant.Light; + switch (EffectiveTheme) { // case SystemThemeOptions.MacOsCatalinaLight: @@ -88,19 +94,21 @@ public SystemThemeOptions Mode // Source = new Uri("avares://AvaloniaStyles/Styles/MacOsBigSurDark.xaml", UriKind.Absolute); // break; case SystemThemeOptions.DarkWindows10: - Source = new Uri("avares://AvaloniaStyles/Styles/Windows10Dark.xaml", UriKind.Absolute); + Source = new Uri("avares://AvaloniaStyles/Styles/Windows10Dark.axaml", UriKind.Absolute); break; case SystemThemeOptions.LightWindows10: - Source = new Uri("avares://AvaloniaStyles/Styles/Windows10Light.xaml", UriKind.Absolute); + Source = new Uri("avares://AvaloniaStyles/Styles/Windows10Light.axaml", UriKind.Absolute); break; case SystemThemeOptions.DarkWindows11: - Source = new Uri("avares://AvaloniaStyles/Styles/Windows11Dark.xaml", UriKind.Absolute); + Source = new Uri("avares://AvaloniaStyles/Styles/Windows11Dark.axaml", UriKind.Absolute); break; case SystemThemeOptions.LightWindows11: - Source = new Uri("avares://AvaloniaStyles/Styles/Windows11Light.xaml", UriKind.Absolute); + Source = new Uri("avares://AvaloniaStyles/Styles/Windows11Light.axaml", UriKind.Absolute); break; } } } + + public static ThemeVariant EffectiveThemeVariant => EffectiveThemeIsDark ? ThemeVariant.Dark : ThemeVariant.Light; } } \ No newline at end of file diff --git a/AvaloniaStyles/Utils/HslColor.cs b/AvaloniaStyles/Utils/HslColor.cs index 01c936427..4f4898933 100644 --- a/AvaloniaStyles/Utils/HslColor.cs +++ b/AvaloniaStyles/Utils/HslColor.cs @@ -47,8 +47,11 @@ public Color ToRgba(double opacity = 1) return new Color((byte)(opacity * 255), (byte)result.X, (byte)result.Y, (byte)result.Z); } - public HslColor Scale(HslDiff scaler) + public HslColor Scale(HslDiff? scaler) { + if (scaler == null) + return this; + double l = L; if (scaler.L > 0.5) { diff --git a/AvaloniaStyles/Utils/SolidBrushTransition.cs b/AvaloniaStyles/Utils/SolidBrushTransition.cs deleted file mode 100644 index fee1c634d..000000000 --- a/AvaloniaStyles/Utils/SolidBrushTransition.cs +++ /dev/null @@ -1,156 +0,0 @@ -using System; -using System.Reactive.Linq; -using Avalonia; -using Avalonia.Animation; -using Avalonia.Animation.Easings; -using Avalonia.Media; -using Avalonia.Reactive; -using Avalonia.Utilities; - -namespace AvaloniaStyles.Utils -{ - public class SolidBrushTransition : Transition - { - /// - public override IObservable DoTransition(IObservable progress, IBrush oldValue, - IBrush newValue) - { - return progress - .Select(p => - { - SolidColorBrush? old = oldValue as SolidColorBrush; - SolidColorBrush? nnew = newValue as SolidColorBrush; - if (old == null || nnew == null) - return newValue; - var f = Easing.Ease(p); - return new SolidColorBrush(new Color( - (byte)((nnew.Color.A - old.Color.A) * f + old.Color.A), - (byte)((nnew.Color.R - old.Color.R) * f + old.Color.R), - (byte)((nnew.Color.G - old.Color.G) * f + old.Color.G), - (byte)((nnew.Color.B - old.Color.B) * f + old.Color.B) - )); - }); - } - } - - // copy paste of Avalonia Transition class, because I do not know why original doesn't work here... - public abstract class Transition : AvaloniaObject, ITransition - { - private AvaloniaProperty? _prop; - - /// - /// Gets or sets the duration of the transition. - /// - public TimeSpan Duration { get; set; } - - /// - /// Gets or sets delay before starting the transition. - /// - public TimeSpan Delay { get; set; } = TimeSpan.Zero; - - /// - /// Gets the easing class to be used. - /// - public Easing Easing { get; set; } = new LinearEasing(); - - /// - public AvaloniaProperty? Property - { - get { return _prop; } - set - { - if (value == null) - return; - - if (!(value.PropertyType.IsAssignableFrom(typeof(T)))) - throw new InvalidCastException - ($"Invalid property type \"{typeof(T).Name}\" for this transition: {GetType().Name}."); - - _prop = value; - } - } - - /// - /// Apply interpolation to the property. - /// - public abstract IObservable DoTransition(IObservable progress, T oldValue, T newValue); - - /// - public virtual IDisposable Apply(Animatable control, IClock clock, object oldValue, object newValue) - { - var transition = DoTransition(new TransitionInstance(clock, Delay, Duration), (T) oldValue, (T) newValue); - return control.Bind((AvaloniaProperty) Property!, transition, Avalonia.Data.BindingPriority.Animation); - } - } - - internal class TransitionInstance : SingleSubscriberObservableBase - { - private IDisposable? _timerSubscription; - private TimeSpan _delay; - private TimeSpan _duration; - private readonly IClock _baseClock; - private IClock? _clock; - - public TransitionInstance(IClock clock, TimeSpan delay, TimeSpan duration) - { - clock = clock ?? throw new ArgumentNullException(nameof(clock)); - - _delay = delay; - _duration = duration; - _baseClock = clock; - } - - private void TimerTick(TimeSpan t) - { - - // [<------------- normalizedTotalDur ------------------>] - // [<---- Delay ---->][<---------- Duration ------------>] - // ^- normalizedDelayEnd - // [<---- normalizedInterpVal --->] - - var normalizedInterpVal = 1d; - - if (!MathUtilities.AreClose(_duration.TotalSeconds, 0d)) - { - var normalizedTotalDur = _delay + _duration; - var normalizedDelayEnd = _delay.TotalSeconds / normalizedTotalDur.TotalSeconds; - var normalizedPresentationTime = t.TotalSeconds / normalizedTotalDur.TotalSeconds; - - if (normalizedPresentationTime < normalizedDelayEnd - || MathUtilities.AreClose(normalizedPresentationTime, normalizedDelayEnd)) - { - normalizedInterpVal = 0d; - } - else - { - normalizedInterpVal = (t.TotalSeconds - _delay.TotalSeconds) / _duration.TotalSeconds; - } - } - - // Clamp interpolation value. - if (normalizedInterpVal >= 1d || normalizedInterpVal < 0d) - { - PublishNext(1d); - PublishCompleted(); - } - else - { - PublishNext(normalizedInterpVal); - } - } - - protected override void Unsubscribed() - { - _timerSubscription?.Dispose(); - if (_clock != null) - _clock.PlayState = PlayState.Stop; - } - - protected override void Subscribed() - { - _clock = new Clock(_baseClock); - _timerSubscription = _clock.Subscribe(TimerTick); - PublishNext(0.0d); - } - } -} \ No newline at end of file diff --git a/AvaloniaStyles/Utils/Win32.cs b/AvaloniaStyles/Utils/Win32.cs index 1414d41c3..91589b6e3 100644 --- a/AvaloniaStyles/Utils/Win32.cs +++ b/AvaloniaStyles/Utils/Win32.cs @@ -1,6 +1,7 @@ using System; using System.Runtime.InteropServices; using Avalonia.Media; +using WDE.Common; namespace AvaloniaStyles.Utils; @@ -56,7 +57,7 @@ public static bool SetDarkMode(IntPtr hwnd, bool on) var msg = Marshal.GetExceptionForHR(result)?.Message; var msg2 = Marshal.GetExceptionForHR(result2)?.Message; - Console.WriteLine($"Can't set window dark mode:\n DWMWA_USE_IMMERSIVE_DARK_MODE method: {msg}\n DWMWA_USE_IMMERSIVE_DARK_MODE_OLD_WINDOWS method: {msg2}"); + LOG.LogWarning($"Can't set window dark mode:\n DWMWA_USE_IMMERSIVE_DARK_MODE method: {msg}\n DWMWA_USE_IMMERSIVE_DARK_MODE_OLD_WINDOWS method: {msg2}"); return false; } @@ -88,7 +89,7 @@ public static bool SetTitleBarColor(IntPtr hwnd, Color color) } var msg = Marshal.GetExceptionForHR(result)?.Message; - Console.WriteLine($"Can't set window caption color: {msg}"); + LOG.LogWarning($"Can't set window caption color: {msg}"); return false; } } \ No newline at end of file diff --git a/BaseDesktopLoader/BaseDesktopLoader.csproj b/BaseDesktopLoader/BaseDesktopLoader.csproj new file mode 100644 index 000000000..56b4b5c80 --- /dev/null +++ b/BaseDesktopLoader/BaseDesktopLoader.csproj @@ -0,0 +1,31 @@ + + + Library + net8.0 + enable + true + + + + ..\bin\$(Configuration)\ + + + + + + + + + + + + + + + + + + + + + diff --git a/BaseDesktopLoader/Program.cs b/BaseDesktopLoader/Program.cs new file mode 100644 index 000000000..57c4aa194 --- /dev/null +++ b/BaseDesktopLoader/Program.cs @@ -0,0 +1,113 @@ +using System.Threading.Tasks; +using System; +using System.IO; +using System.Reflection; +using System.Runtime.InteropServices; +using AsyncAwaitBestPractices; +using Avalonia; +using Avalonia.ReactiveUI; +using Avalonia.Threading; +using Projektanker.Icons.Avalonia; +using Projektanker.Icons.Avalonia.MaterialDesign; +using WDE.Common; +using WDE.Common.Tasks; +using WoWDatabaseEditorCore; +using WoWDatabaseEditorCore.Avalonia; +using WoWDatabaseEditorCore.Avalonia.Utils; +using WoWDatabaseEditorCore.Managers; +using WoWDatabaseEditorCore.Services.FileSystemService; + +namespace BaseDesktopLoader +{ + public abstract class BaseProgramLoader + { + // Initialization code. Don't use any Avalonia, third-party APIs or any + // SynchronizationContext-reliant code before AppMain is called: things aren't initialized + // yet and stuff might break. + public static void Main(System.Type[] modules, string[] args) + { + MigrateMacOsSettings(); + + WoWDatabaseEditorCore.Avalonia.Program.PreloadedModules = modules; + + if (!BeforeBuildApp(args)) + return; + + var app = BuildAvaloniaApp(); + try + { + app.StartWithClassicDesktopLifetime(args); + } + catch (Exception e) + { + FatalErrorHandler.ExceptionOccured(e, args); + } + TheEngine.TheEngine.Deinit(); + } + + private static void MigrateMacOsSettings() + { + if (RuntimeInformation.IsOSPlatform(OSPlatform.OSX)) + { + var localDataPath = Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData); + var oldLocalDataPath = Path.Join(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".local", "share"); + + var newSettingsPath = Path.Join(localDataPath, FileSystem.APPLICATION_FOLDER); + var oldSettingsPath = Path.Join(oldLocalDataPath, FileSystem.APPLICATION_FOLDER); + + if (Directory.Exists(oldSettingsPath) && !Directory.Exists(newSettingsPath)) + { + Directory.Move(oldSettingsPath, newSettingsPath); + } + } + } + + public static bool BeforeBuildApp(string[] args) + { + FixCurrentDirectory(); + if (ProgramBootstrap.TryLaunchUpdaterIfNeeded(args)) + return false; + System.Threading.Thread.CurrentThread.CurrentCulture = System.Globalization.CultureInfo.InvariantCulture; + SafeFireAndForgetExtensions.SetDefaultExceptionHandling(x => LOG.LogError(x)); + TaskScheduler.UnobservedTaskException += (sender, eventArgs) => LOG.LogError(eventArgs.Exception); + GlobalApplication.Arguments.Init(args); + + if (GlobalApplication.Arguments.IsArgumentSet("console")) + { + if (OperatingSystem.IsWindows()) + { + if (!Win32.AttachConsole(Win32.ATTACH_PARENT_PROCESS)) + Win32.AllocConsole(); + } + } + + return true; + } + + private static void FixCurrentDirectory() + { + #pragma warning disable IL3000 + var path = Assembly.GetExecutingAssembly().Location; + #pragma warning restore IL3000 + if (string.IsNullOrEmpty(path)) + path = System.AppContext.BaseDirectory; + var exePath = new FileInfo(path); + if (exePath.Directory != null) + Directory.SetCurrentDirectory(exePath.Directory.FullName); + } + + // Avalonia configuration, don't remove; also used by visual designer. + public static AppBuilder BuildAvaloniaApp() + { + IconProvider.Current + .Register(); + + var configuration = AppBuilder.Configure() + .UsePlatformDetect() + .UseReactiveUI() + .LogToTrace(); + + return configuration; + } + } +} diff --git a/DBCD b/DBCD index e79a6456e..f19c8588a 160000 --- a/DBCD +++ b/DBCD @@ -1 +1 @@ -Subproject commit e79a6456e828a4562f05df961a969cc147122097 +Subproject commit f19c8588ae0e021fe917188108b2c9e1f3d2a75d diff --git a/DatabaseTester/ConsoleMessageBoxService.cs b/DatabaseTester/ConsoleMessageBoxService.cs index ae039d0c9..73e591528 100644 --- a/DatabaseTester/ConsoleMessageBoxService.cs +++ b/DatabaseTester/ConsoleMessageBoxService.cs @@ -7,8 +7,5 @@ public class ConsoleMessageBoxService : IMessageBoxService public Task ShowDialog(IMessageBox messageBox) { throw new Exception(messageBox.MainInstruction + messageBox.Content); - Console.WriteLine(messageBox.MainInstruction); - Console.WriteLine(messageBox.Content); - return Task.FromException(new Exception(messageBox.Content)); } } \ No newline at end of file diff --git a/DatabaseTester/DatabaseTester.csproj b/DatabaseTester/DatabaseTester.csproj index c6dd2613b..55a079547 100644 --- a/DatabaseTester/DatabaseTester.csproj +++ b/DatabaseTester/DatabaseTester.csproj @@ -2,7 +2,7 @@ Exe - net7.0 + net8.0 enable enable @@ -22,7 +22,7 @@ - + diff --git a/DatabaseTester/Program.cs b/DatabaseTester/Program.cs index bae8b9a08..b7d811764 100644 --- a/DatabaseTester/Program.cs +++ b/DatabaseTester/Program.cs @@ -65,7 +65,7 @@ private static bool TryGetDefaultObjectForType(Type type, out object? obj) { obj = null; if (type == typeof(string)) - obj = ""; + obj = "a"; else if (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Nullable<>) && TryGetDefaultObjectForType(type.GetGenericArguments()[0], out var innerNullable)) @@ -299,8 +299,8 @@ public static async Task Test(string[] args, params ICoreVersion[] cores } // methods with > 1 parameters - worldDb.GetScriptFor(0, 0, SmartScriptType.Creature); - worldDb.GetConditionsFor(0, 0, 0); + await worldDb.GetScriptForAsync(0, 0, SmartScriptType.Creature); + await worldDb.GetConditionsForAsync(0, 0, 0); await worldDb.GetConditionsForAsync(IDatabaseProvider.ConditionKeyMask.All, new IDatabaseProvider.ConditionKey(0, 0, 0, 0)); await worldDb.GetConditionsForAsync(IDatabaseProvider.ConditionKeyMask.All, new List(){new IDatabaseProvider.ConditionKey(0, 0, 0, 0)}); await worldDb.FindSmartScriptLinesBy(new (IDatabaseProvider.SmartLinePropertyType what, int whatValue, int parameterIndex, long valueToSearch)[] { (IDatabaseProvider.SmartLinePropertyType.Action, 0, 0, 0) }); diff --git a/Directory.Build.props b/Directory.Build.props index 3dcbf076d..84bcc198d 100644 --- a/Directory.Build.props +++ b/Directory.Build.props @@ -2,10 +2,22 @@ - https://nuget-feed-all.avaloniaui.net/v3/index.json + https://nuget-feed-all.avaloniaui.net/v3/index.json;https://www.myget.org/F/xamlbehaviors-nightly/api/v3/index.json CS0067,CS3021,CS1998,CA1416,CS1591 - 0.10.999-cibuild0044575-beta + $(NoWarn),CS0649 + $(NoWarn),NUnit2005,NUnit2049 + $(NoWarn),CS0436 + + true + CS0414 + $(WarningsNotAsErrors),CS0169 + $(WarningsNotAsErrors),NUnit2005 + $(WarningsNotAsErrors),CS0436 + $(WarningsNotAsErrors),NU5104 + $(WarningsNotAsErrors),CS1574 + + $(WarningsAsErrors),AVP1000, AVP1001, AVP1002, AVP1010, AVP1011, AVP1012, AVP1013, AVP1020, AVP1021, AVP1022, AVP1030, AVP1031, AVP1032, AVP1040, AVA2001 preview @@ -19,4 +31,4 @@ $(OutputPath) $(SolutionDir)bin\doc\$(Configuration)\$(MSBuildProjectName).xml - \ No newline at end of file + diff --git a/Dock b/Dock index 867ffea9f..77017d455 160000 --- a/Dock +++ b/Dock @@ -1 +1 @@ -Subproject commit 867ffea9f5f4ca337f6942e6c9aaf87931de4d32 +Subproject commit 77017d455593d4ae1797cf90a5e298431a414fe4 diff --git a/GeminiGraphEditor/GeminiGraphEditor.csproj b/GeminiGraphEditor/GeminiGraphEditor.csproj index 648893482..5cdb11394 100644 --- a/GeminiGraphEditor/GeminiGraphEditor.csproj +++ b/GeminiGraphEditor/GeminiGraphEditor.csproj @@ -1,6 +1,6 @@  - net7.0-windows7.0 + net8.0-windows7.0 Library false true diff --git a/LoaderAvalonia.Web/AppBundle/Logo.svg b/LoaderAvalonia.Web/AppBundle/Logo.svg new file mode 100644 index 000000000..9685a23af --- /dev/null +++ b/LoaderAvalonia.Web/AppBundle/Logo.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/LoaderAvalonia.Web/AppBundle/app.css b/LoaderAvalonia.Web/AppBundle/app.css new file mode 100644 index 000000000..a424538ba --- /dev/null +++ b/LoaderAvalonia.Web/AppBundle/app.css @@ -0,0 +1,74 @@ +:root { + --sat: env(safe-area-inset-top); + --sar: env(safe-area-inset-right); + --sab: env(safe-area-inset-bottom); + --sal: env(safe-area-inset-left); +} + +/* HTML styles for the splash screen */ + +.highlight { + color: white; + font-size: 2.5rem; + display: block; +} + +.purple { + color: #8b44ac; +} + +.icon { + opacity: 0.05; + height: 35%; + width: 35%; + position: absolute; + background-repeat: no-repeat; + right: 0px; + bottom: 0px; + margin-right: 3%; + margin-bottom: 5%; + z-index: 5000; + background-position: right bottom; + pointer-events: none; +} + +#avalonia-splash a { + color: whitesmoke; + text-decoration: none; +} + +.center { + display: flex; + justify-content: center; + align-items: center; + height: 100vh; +} + +#avalonia-splash { + position: relative; + height: 100%; + width: 100%; + color: whitesmoke; + background: #1b2a4e; + font-family: 'Nunito', sans-serif; + background-position: center; + background-size: cover; + background-repeat: no-repeat; + justify-content: center; + align-items: center; +} + +.splash-close { + animation: fadeout 0.25s linear forwards; +} + +@keyframes fadeout { + 0% { + opacity: 100%; + } + + 100% { + opacity: 0; + visibility: collapse; + } +} diff --git a/LoaderAvalonia.Web/AppBundle/favicon.ico b/LoaderAvalonia.Web/AppBundle/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..da8d49ff9b94e52778f5324a1b87dd443a698b57 GIT binary patch literal 176111 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 zds⪻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

r@R8DskCg$QY%d=x6zD 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

!LxQh!NmF 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 + + + + WoW Database Edtor Online Demo Edition + + + + + + + + + + +
+
+
+

+ Powered by + Avalonia UI +

+
+ Avalonia Logo +
+
+ + + + \ No newline at end of file diff --git a/LoaderAvalonia.Web/AppBundle/main.js b/LoaderAvalonia.Web/AppBundle/main.js new file mode 100644 index 000000000..599c341eb --- /dev/null +++ b/LoaderAvalonia.Web/AppBundle/main.js @@ -0,0 +1,31 @@ +import { dotnet } from './_framework/dotnet.js' + +const is_browser = typeof window != "undefined"; +if (!is_browser) throw new Error(`Expected to be running in a browser`); + +const dotnetRuntime = await dotnet + .withDiagnosticTracing(false) + .withApplicationArgumentsFromQuery() + .create(); + +dotnetRuntime.setModuleImports("main.js", { + window: { + history: { + replaceState: (title, href) => globalThis.window.history.replaceState(null, title, href) + } + }, + localStorage: { + getValue: (key) => globalThis.localStorage.getItem(key), + setValue: (key, value) => globalThis.localStorage.setItem(key, value) + } +}); + +const config = dotnetRuntime.getConfig(); + +let href = window.location.href; +if (href.indexOf("?") !== -1) + href = href.slice(0, href.indexOf("?")) +if (href.indexOf("index.html") !== -1) + href = href.slice(0, href.indexOf("index.html")) + +await dotnetRuntime.runMain(config.mainAssemblyName, ["--search", window.location.search, "--href", href]); \ No newline at end of file diff --git a/LoaderAvalonia.Web/LoaderAvalonia.Web.csproj b/LoaderAvalonia.Web/LoaderAvalonia.Web.csproj new file mode 100644 index 000000000..663dd16fc --- /dev/null +++ b/LoaderAvalonia.Web/LoaderAvalonia.Web.csproj @@ -0,0 +1,116 @@ + + + net8.0-browser + browser-wasm + AppBundle\main.js + Exe + enable + nullable + false + 1073741824 + true + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/LoaderAvalonia.Web/NullDebuggerService.cs b/LoaderAvalonia.Web/NullDebuggerService.cs new file mode 100644 index 000000000..2a0ff82a2 --- /dev/null +++ b/LoaderAvalonia.Web/NullDebuggerService.cs @@ -0,0 +1,82 @@ +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using WDE.Common.Debugging; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.Web; + +[AutoRegister] +[SingleInstance] +public class NullDebuggerService : IDebuggerService +{ + public DebugPointId CreateDebugPoint(IDebugPointSource source, IDebugPointPayload payload) => default; + + public void ScheduleRemoveDebugPoint(DebugPointId id) { } + + public async Task RemoveDebugPointAsync(DebugPointId id, bool remoteAlreadyRemoved = false) { } + + public async Task ClearAllDebugPoints() { } + + public bool TryFindDebugPoint(Func matcher, out DebugPointId id) where T : IDebugPointPayload + { + id = default; + return false; + } + + public async Task RemoveIfAsync(Func matcher, bool remoteAlreadyRemoved = false) where T : IDebugPointPayload { } + + public bool TryGetPayload(DebugPointId id, out T payload) where T : IDebugPointPayload + { + payload = default!; + return false; + } + + public void SetLog(DebugPointId point, string? path) { } + + public void SetEnabled(DebugPointId point, bool enabled) { } + + public void SetActivated(DebugPointId point, bool activated) { } + + public void SetSuspendExecution(DebugPointId point, bool suspend) { } + + public void SetGenerateStacktrace(DebugPointId point, bool generate) { } + + public void SetPayload(DebugPointId id, IDebugPointPayload payload) { } + + public void SetLogCondition(DebugPointId point, string? condition) { } + + public void ForceSetSynchronized(DebugPointId point) { } + + public void ForceSetPending(DebugPointId point) { } + + public string? GetLogFormat(DebugPointId point) => null; + + public string? GetLogCondition(DebugPointId point) => null; + + public bool GetActivated(DebugPointId point) => false; + + public bool GetEnabled(DebugPointId point) => false; + + public bool GetSuspendExecution(DebugPointId point) => false; + + public bool GetGenerateStacktrace(DebugPointId point) => false; + + public bool IsBreakpointHit(DebugPointId point) => false; + + public IDebugPointSource GetSource(DebugPointId point) => null!; + + public bool HasDebugPoint(DebugPointId id) => false; + + public async Task Synchronize(DebugPointId id) { } + + public BreakpointState GetState(DebugPointId breakpoint) => BreakpointState.Pending; + + public bool IsConnected => false; + public IReadOnlyList Sources { get; set; } = Array.Empty(); + public IEnumerable DebugPoints { get; set; } = Array.Empty(); + public event Action? DebugPointAdded; + public event Action? DebugPointRemoving; + public event Action? DebugPointRemoved; + public event Action? DebugPointChanged; +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/NullGameView.cs b/LoaderAvalonia.Web/NullGameView.cs new file mode 100644 index 000000000..bcaef92bd --- /dev/null +++ b/LoaderAvalonia.Web/NullGameView.cs @@ -0,0 +1,16 @@ +using WDE.Common.Services; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.Web; + +[AutoRegister] +[SingleInstance] +public class NullGameView : IGameViewService +{ + public bool IsSupported => false; + + public void Open() + { + + } +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/NullRemoteConnectorService.cs b/LoaderAvalonia.Web/NullRemoteConnectorService.cs new file mode 100644 index 000000000..2c0ae3ce2 --- /dev/null +++ b/LoaderAvalonia.Web/NullRemoteConnectorService.cs @@ -0,0 +1,36 @@ +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using WDE.Common.Services; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.Web; + +[AutoRegister] +[SingleInstance] +public class NullRemoteConnectorService : IRemoteConnectorService +{ + public event Action? OnLog; + + public bool IsConnected => false; + + public bool HasValidSettings => true; + + public async Task ExecuteCommand(IRemoteCommand command) + { + return ""; + } + + public async Task ExecuteCommands(IList commands) + { + } + + public event Action? EditorCommandReceived; + public event Action? EditorConnected; + public event Action? EditorDisconnected; + + public IList Merge(IList commands) + { + return commands; + } +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/NullSourceCodePathService.cs b/LoaderAvalonia.Web/NullSourceCodePathService.cs new file mode 100644 index 000000000..9cd322f2a --- /dev/null +++ b/LoaderAvalonia.Web/NullSourceCodePathService.cs @@ -0,0 +1,13 @@ +using System; +using System.Collections.Generic; +using WDE.Common.Services; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.Web; + +[AutoRegister] +[SingleInstance] +public class NullSourceCodePathService : ISourceCodePathService +{ + public IReadOnlyList SourceCodePaths { get; set; } = Array.Empty(); +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/NullUpdateViewModel.cs b/LoaderAvalonia.Web/NullUpdateViewModel.cs new file mode 100644 index 000000000..c99ff35be --- /dev/null +++ b/LoaderAvalonia.Web/NullUpdateViewModel.cs @@ -0,0 +1,16 @@ +using System.ComponentModel; +using System.Windows.Input; +using WDE.Common.Services; +using WDE.Common.Utils; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.Web; + +[AutoRegister] +[SingleInstance] +public class NullUpdateViewModel : IUpdateViewModel +{ + public event PropertyChangedEventHandler? PropertyChanged; + public bool HasPendingUpdate => false; + public ICommand InstallPendingCommandUpdate { get; } = AlwaysDisabledCommand.Command; +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/NullVisualStudioManagerViewModel.cs b/LoaderAvalonia.Web/NullVisualStudioManagerViewModel.cs new file mode 100644 index 000000000..b5c31a80c --- /dev/null +++ b/LoaderAvalonia.Web/NullVisualStudioManagerViewModel.cs @@ -0,0 +1,11 @@ +using WDE.Common.Managers; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.Web; + +[AutoRegister] +[SingleInstance] +public class NullVisualStudioManagerViewModel : IVisualStudioManagerViewModel +{ + +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/Program.cs b/LoaderAvalonia.Web/Program.cs new file mode 100644 index 000000000..709bf448f --- /dev/null +++ b/LoaderAvalonia.Web/Program.cs @@ -0,0 +1,77 @@ +using System.Runtime.Versioning; +using System.Threading.Tasks; +using Avalonia; +using Avalonia.Browser; +using Avalonia.ReactiveUI; +using WDE.Common.Avalonia; +using WDE.Common.Tasks; +using WDE.CommonViews.Avalonia; +using WDE.Conditions; +using WDE.DatabaseEditors; +using WDE.DatabaseEditors.Avalonia; +using WDE.DbcStore; +using WDE.EventAiEditor.Avalonia; +using WDE.EventScriptsEditor; +using WDE.History; +using WDE.HttpDatabase; +using WDE.LootEditor; +using WDE.MySqlDatabaseCommon; +using WDE.Parameters; +using WDE.Profiles; +using WDE.QueryGenerators; +using WDE.Sessions; +using WDE.SmartScriptEditor.Avalonia; +using WDE.Solutions; +using WDE.Spells; +using WDE.SQLEditor; +using WDE.SqlInterpreter; +using WDE.Trinity; +using WDE.TrinitySmartScriptEditor; +using WoWDatabaseEditorCore.Avalonia; + +[assembly: SupportedOSPlatform("browser")] + +namespace LoaderAvalonia.Web; + +internal partial class Program +{ + static System.Type[] modules = new System.Type[] + { + typeof(CommonAvaloniaModule), + typeof(CommonViewsModule), + typeof(ConditionsModule), + typeof(DatabaseEditorsModule), + typeof(DatabaseEditorsAvaloniaModule), + typeof(MySqlDatabaseCommonModule), + typeof(ParametersModule), + typeof(SmartScriptAvaloniaModule), + typeof(SpellsModule), + typeof(SqlInterpreterModule), + typeof(DbcStoreModule), + typeof(HistoryModule), + typeof(EventAiAvaloniaModule), + typeof(SessionsModule), + typeof(SolutionsModule), + typeof(QueryGeneratorModule), + typeof(TrinityModule), + typeof(SmartScriptModule), + typeof(WebModule), + typeof(LootEditorModule), + typeof(EventScriptsModule), + typeof(HttpDatabaseModule), + typeof(ProfilesModule), + typeof(SqlEditorModule) + }; + + private static async Task Main(string[] args) + { + GlobalApplication.Arguments.Init(args); + await BuildAvaloniaApp() + .WithInterFont() + .UseReactiveUI() + .StartBrowserAppAsync("out"); + } + + public static AppBuilder BuildAvaloniaApp() + => AppBuilder.Configure(); +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/Properties/launchSettings.json b/LoaderAvalonia.Web/Properties/launchSettings.json new file mode 100644 index 000000000..3104f57fd --- /dev/null +++ b/LoaderAvalonia.Web/Properties/launchSettings.json @@ -0,0 +1,13 @@ +{ + "profiles": { + "WoWDatabaseEditorDemo": { + "commandName": "Project", + "launchBrowser": false, + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + }, + "applicationUrl": "https://localhost:5001;http://localhost:5000", + "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/debug?browser={browserInspectUri}" + } + } +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/WebModule.cs b/LoaderAvalonia.Web/WebModule.cs new file mode 100644 index 000000000..9bcd87976 --- /dev/null +++ b/LoaderAvalonia.Web/WebModule.cs @@ -0,0 +1,8 @@ +using WDE.Module; + +namespace LoaderAvalonia.Web; + +public class WebModule : ModuleBase +{ + +} \ No newline at end of file diff --git a/LoaderAvalonia.Web/runtimeconfig.template.json b/LoaderAvalonia.Web/runtimeconfig.template.json new file mode 100644 index 000000000..c6990ba79 --- /dev/null +++ b/LoaderAvalonia.Web/runtimeconfig.template.json @@ -0,0 +1,11 @@ +{ + "wasmHostProperties": { + "perHostConfig": [ + { + "name": "browser", + "html-path": "index.html", + "Host": "browser" + } + ] + } +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/AppDelegate.cs b/LoaderAvalonia.iOS/AppDelegate.cs new file mode 100644 index 000000000..821de7ab1 --- /dev/null +++ b/LoaderAvalonia.iOS/AppDelegate.cs @@ -0,0 +1,20 @@ +using Avalonia; +using Avalonia.iOS; +using Avalonia.ReactiveUI; +using Foundation; +using WoWDatabaseEditorCore.Avalonia; + +namespace LoaderAvalonia.iOS; + +// The UIApplicationDelegate for the application. This class is responsible for launching the +// User Interface of the application, as well as listening (and optionally responding) to +// application events from iOS. +[Register("AppDelegate")] +public partial class AppDelegate : AvaloniaAppDelegate +{ + protected override AppBuilder CustomizeAppBuilder(AppBuilder builder) + { + return base.CustomizeAppBuilder(builder) + .WithInterFont(); + } +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/Entitlements.plist b/LoaderAvalonia.iOS/Entitlements.plist new file mode 100644 index 000000000..0c67376eb --- /dev/null +++ b/LoaderAvalonia.iOS/Entitlements.plist @@ -0,0 +1,5 @@ + + + + + diff --git a/LoaderAvalonia.iOS/Info.plist b/LoaderAvalonia.iOS/Info.plist new file mode 100644 index 000000000..c28c6c009 --- /dev/null +++ b/LoaderAvalonia.iOS/Info.plist @@ -0,0 +1,47 @@ + + + + + CFBundleDisplayName + MyApp + CFBundleIdentifier + x.mytestapp + CFBundleShortVersionString + 1.0 + CFBundleVersion + 36 + LSRequiresIPhoneOS + + MinimumOSVersion + 13.0 + UIDeviceFamily + + 1 + 2 + + UILaunchStoryboardName + LaunchScreen + UIRequiredDeviceCapabilities + + armv7 + + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UIStatusBarHidden + + UIViewControllerBasedStatusBarAppearance + + + diff --git a/LoaderAvalonia.iOS/LoaderAvalonia.iOS.csproj b/LoaderAvalonia.iOS/LoaderAvalonia.iOS.csproj new file mode 100644 index 000000000..9af42456f --- /dev/null +++ b/LoaderAvalonia.iOS/LoaderAvalonia.iOS.csproj @@ -0,0 +1,51 @@ + + + Exe + net8.0-ios + 13.0 + enable + nullable + false + + + + + Apple Development: b.andy@windowslive.com (Q5L623R3A3) + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/LoaderAvalonia.iOS/Main.cs b/LoaderAvalonia.iOS/Main.cs new file mode 100644 index 000000000..9405e7dd1 --- /dev/null +++ b/LoaderAvalonia.iOS/Main.cs @@ -0,0 +1,69 @@ +using UIKit; +using WDE.Common.Avalonia; +using WDE.Common.Tasks; +using WDE.CommonViews.Avalonia; +using WDE.Conditions; +using WDE.DatabaseEditors; +using WDE.DatabaseEditors.Avalonia; +using WDE.DbcStore; +using WDE.EventAiEditor.Avalonia; +using WDE.EventScriptsEditor; +using WDE.History; +using WDE.HttpDatabase; +using WDE.LootEditor; +using WDE.MySqlDatabaseCommon; +using WDE.Parameters; +using WDE.Profiles; +using WDE.QueryGenerators; +using WDE.Sessions; +using WDE.SmartScriptEditor.Avalonia; +using WDE.Solutions; +using WDE.Spells; +using WDE.SQLEditor; +using WDE.SqlInterpreter; +using WDE.Trinity; +using WDE.TrinitySmartScriptEditor; + +namespace LoaderAvalonia.iOS; + +public class Application +{ + + static System.Type[] modules = new System.Type[] + { + typeof(iOSModule), + typeof(CommonAvaloniaModule), + typeof(CommonViewsModule), + typeof(ConditionsModule), + typeof(DatabaseEditorsModule), + typeof(DatabaseEditorsAvaloniaModule), + typeof(MySqlDatabaseCommonModule), + typeof(ParametersModule), + typeof(SmartScriptAvaloniaModule), + typeof(SpellsModule), + typeof(SqlInterpreterModule), + typeof(DbcStoreModule), + typeof(HistoryModule), + typeof(EventAiAvaloniaModule), + typeof(SessionsModule), + typeof(SolutionsModule), + typeof(QueryGeneratorModule), + typeof(TrinityModule), + typeof(SmartScriptModule), + typeof(LootEditorModule), + typeof(EventScriptsModule), + typeof(HttpDatabaseModule), + typeof(ProfilesModule), + typeof(SqlEditorModule) + }; + + // This is the main entry point of the application. + static void Main(string[] args) + { + WoWDatabaseEditorCore.Avalonia.Program.PreloadedModules = modules; + GlobalApplication.Arguments.Init(new[]{"--href", "http://192.168.1.106:5000"}); + // if you want to use a different Application Delegate class from "AppDelegate" + // you can specify it here. + UIApplication.Main(args, null, typeof(AppDelegate)); + } +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/NullDebuggerService.cs b/LoaderAvalonia.iOS/NullDebuggerService.cs new file mode 100644 index 000000000..f37b6be22 --- /dev/null +++ b/LoaderAvalonia.iOS/NullDebuggerService.cs @@ -0,0 +1,82 @@ +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using WDE.Common.Debugging; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.iOS; + +[AutoRegister] +[SingleInstance] +public class NullDebuggerService : IDebuggerService +{ + public DebugPointId CreateDebugPoint(IDebugPointSource source, IDebugPointPayload payload) => default; + + public void ScheduleRemoveDebugPoint(DebugPointId id) { } + + public async Task RemoveDebugPointAsync(DebugPointId id, bool remoteAlreadyRemoved = false) { } + + public async Task ClearAllDebugPoints() { } + + public bool TryFindDebugPoint(Func matcher, out DebugPointId id) where T : IDebugPointPayload + { + id = default; + return false; + } + + public async Task RemoveIfAsync(Func matcher, bool remoteAlreadyRemoved = false) where T : IDebugPointPayload { } + + public bool TryGetPayload(DebugPointId id, out T payload) where T : IDebugPointPayload + { + payload = default!; + return false; + } + + public void SetLog(DebugPointId point, string? path) { } + + public void SetEnabled(DebugPointId point, bool enabled) { } + + public void SetActivated(DebugPointId point, bool activated) { } + + public void SetSuspendExecution(DebugPointId point, bool suspend) { } + + public void SetGenerateStacktrace(DebugPointId point, bool generate) { } + + public void SetPayload(DebugPointId id, IDebugPointPayload payload) { } + + public void SetLogCondition(DebugPointId point, string? condition) { } + + public void ForceSetSynchronized(DebugPointId point) { } + + public void ForceSetPending(DebugPointId point) { } + + public string? GetLogFormat(DebugPointId point) => null; + + public string? GetLogCondition(DebugPointId point) => null; + + public bool GetActivated(DebugPointId point) => false; + + public bool GetEnabled(DebugPointId point) => false; + + public bool GetSuspendExecution(DebugPointId point) => false; + + public bool GetGenerateStacktrace(DebugPointId point) => false; + + public bool IsBreakpointHit(DebugPointId point) => false; + + public IDebugPointSource GetSource(DebugPointId point) => null!; + + public bool HasDebugPoint(DebugPointId id) => false; + + public async Task Synchronize(DebugPointId id) { } + + public BreakpointState GetState(DebugPointId breakpoint) => BreakpointState.Pending; + + public bool IsConnected => false; + public IReadOnlyList Sources { get; set; } = Array.Empty(); + public IEnumerable DebugPoints { get; set; } = Array.Empty(); + public event Action? DebugPointAdded; + public event Action? DebugPointRemoving; + public event Action? DebugPointRemoved; + public event Action? DebugPointChanged; +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/NullGameView.cs b/LoaderAvalonia.iOS/NullGameView.cs new file mode 100644 index 000000000..646f3143f --- /dev/null +++ b/LoaderAvalonia.iOS/NullGameView.cs @@ -0,0 +1,16 @@ +using WDE.Common.Services; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.iOS; + +[AutoRegister] +[SingleInstance] +public class NullGameView : IGameViewService +{ + public bool IsSupported => false; + + public void Open() + { + + } +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/NullRemoteConnectorService.cs b/LoaderAvalonia.iOS/NullRemoteConnectorService.cs new file mode 100644 index 000000000..fce263e45 --- /dev/null +++ b/LoaderAvalonia.iOS/NullRemoteConnectorService.cs @@ -0,0 +1,36 @@ +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using WDE.Common.Services; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.iOS; + +[AutoRegister] +[SingleInstance] +public class NullRemoteConnectorService : IRemoteConnectorService +{ + public event Action? OnLog; + + public bool IsConnected => false; + + public bool HasValidSettings => true; + + public async Task ExecuteCommand(IRemoteCommand command) + { + return ""; + } + + public async Task ExecuteCommands(IList commands) + { + } + + public event Action? EditorCommandReceived; + public event Action? EditorConnected; + public event Action? EditorDisconnected; + + public IList Merge(IList commands) + { + return commands; + } +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/NullSourceCodePathService.cs b/LoaderAvalonia.iOS/NullSourceCodePathService.cs new file mode 100644 index 000000000..e53d2f464 --- /dev/null +++ b/LoaderAvalonia.iOS/NullSourceCodePathService.cs @@ -0,0 +1,13 @@ +using System; +using System.Collections.Generic; +using WDE.Common.Services; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.iOS; + +[AutoRegister] +[SingleInstance] +public class NullSourceCodePathService : ISourceCodePathService +{ + public IReadOnlyList SourceCodePaths { get; set; } = Array.Empty(); +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/NullUpdateViewModel.cs b/LoaderAvalonia.iOS/NullUpdateViewModel.cs new file mode 100644 index 000000000..03ff6d1e7 --- /dev/null +++ b/LoaderAvalonia.iOS/NullUpdateViewModel.cs @@ -0,0 +1,16 @@ +using System.ComponentModel; +using System.Windows.Input; +using WDE.Common.Services; +using WDE.Common.Utils; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.iOS; + +[AutoRegister] +[SingleInstance] +public class NullUpdateViewModel : IUpdateViewModel +{ + public event PropertyChangedEventHandler? PropertyChanged; + public bool HasPendingUpdate => false; + public ICommand InstallPendingCommandUpdate { get; } = AlwaysDisabledCommand.Command; +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/NullVisualStudioManagerViewModel.cs b/LoaderAvalonia.iOS/NullVisualStudioManagerViewModel.cs new file mode 100644 index 000000000..aaf49c484 --- /dev/null +++ b/LoaderAvalonia.iOS/NullVisualStudioManagerViewModel.cs @@ -0,0 +1,11 @@ +using WDE.Common.Managers; +using WDE.Module.Attributes; + +namespace LoaderAvalonia.iOS; + +[AutoRegister] +[SingleInstance] +public class NullVisualStudioManagerViewModel : IVisualStudioManagerViewModel +{ + +} \ No newline at end of file diff --git a/LoaderAvalonia.iOS/Resources/LaunchScreen.xib b/LoaderAvalonia.iOS/Resources/LaunchScreen.xib new file mode 100644 index 000000000..4e33f2f48 --- /dev/null +++ b/LoaderAvalonia.iOS/Resources/LaunchScreen.xib @@ -0,0 +1,43 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/LoaderAvalonia.iOS/iOSModule.cs b/LoaderAvalonia.iOS/iOSModule.cs new file mode 100644 index 000000000..0db37a824 --- /dev/null +++ b/LoaderAvalonia.iOS/iOSModule.cs @@ -0,0 +1,8 @@ +using WDE.Module; + +namespace LoaderAvalonia.iOS; + +public class iOSModule : ModuleBase +{ + +} \ No newline at end of file diff --git a/LoaderAvalonia/LoaderAvalonia.csproj b/LoaderAvalonia/LoaderAvalonia.csproj index 57612d293..605a23c1b 100644 --- a/LoaderAvalonia/LoaderAvalonia.csproj +++ b/LoaderAvalonia/LoaderAvalonia.csproj @@ -2,24 +2,30 @@ WinExe - net7.0 + net8.0 Debug;Release AnyCPU ../WoWDatabaseEditorCore.Avalonia/Icon.ico - WoWDatabaseEditorCore.Avalonia + WoWDatabaseEditor false ..\bin\$(Configuration)\ - + + + + + + app.manifest + diff --git a/LoaderAvalonia/Program.cs b/LoaderAvalonia/Program.cs index 11f064679..ba641b8db 100644 --- a/LoaderAvalonia/Program.cs +++ b/LoaderAvalonia/Program.cs @@ -1,3 +1,8 @@ +using Avalonia; +using Avalonia.ReactiveUI; +using BaseDesktopLoader; +using Projektanker.Icons.Avalonia; +using Projektanker.Icons.Avalonia.MaterialDesign; using WDE.AzerothCore; using WDE.CMaNGOS; using WDE.CMMySqlDatabase; @@ -39,10 +44,10 @@ using WDE.PathPreviewTool; using WDE.FirstTimeWizard; using WDE.LootEditor; -using WDE.LootEditor.Models; using WDE.QueryGenerators; using WDE.Profiles; using WDE.SqlWorkbench; +using WoWDatabaseEditorCore.Avalonia; namespace LoaderAvalonia { @@ -97,8 +102,18 @@ public static void Main(string[] args) typeof(SqlWorkbenchModule), typeof(DebuggerModule) }; - WoWDatabaseEditorCore.Avalonia.Program.PreloadedModules = modules; - WoWDatabaseEditorCore.Avalonia.Program.Main(args); + BaseProgramLoader.Main(modules, args); + } + + // Avalonia configuration, don't remove; only used by visual designer. + public static AppBuilder BuildAvaloniaApp() + { + var configuration = AppBuilder.Configure() + .UsePlatformDetect() + .UseReactiveUI() + .LogToTrace(); + + return configuration; } } } \ No newline at end of file diff --git a/Modules/AvaloniaGraph/AvaloniaGraph.csproj b/Modules/AvaloniaGraph/AvaloniaGraph.csproj index 9ac8d398d..d0ec9b0f6 100644 --- a/Modules/AvaloniaGraph/AvaloniaGraph.csproj +++ b/Modules/AvaloniaGraph/AvaloniaGraph.csproj @@ -1,10 +1,10 @@ - net7.0 + net8.0 Debug;Release AnyCPU enable - nullable + $(WarningsAsErrors),nullable diff --git a/Modules/AvaloniaGraph/Controls/BezierLine.cs b/Modules/AvaloniaGraph/Controls/BezierLine.cs index db392ec27..448c224cc 100644 --- a/Modules/AvaloniaGraph/Controls/BezierLine.cs +++ b/Modules/AvaloniaGraph/Controls/BezierLine.cs @@ -11,7 +11,7 @@ public class BezierLine : Shape //, ICustomHitTest AvaloniaProperty.Register(nameof(StartPoint)); public static readonly StyledProperty EndPointProperty = - AvaloniaProperty.Register(nameof(StartPoint)); + AvaloniaProperty.Register(nameof(EndPoint)); static BezierLine() { @@ -74,20 +74,18 @@ protected virtual PathSegments GetSegments() protected override Geometry? CreateDefiningGeometry() { + var figure = new PathFigure + { + IsFilled = false, + StartPoint = Relative + ? StartPoint - new Point(Math.Min(StartPoint.X, EndPoint.X), Math.Min(StartPoint.Y, EndPoint.Y)) + : StartPoint, + Segments = GetSegments(), + IsClosed = false + }; return new PathGeometry { - Figures = - { - new PathFigure - { - IsFilled = false, - StartPoint = Relative - ? StartPoint - new Point(Math.Min(StartPoint.X, EndPoint.X), Math.Min(StartPoint.Y, EndPoint.Y)) - : StartPoint, - Segments = GetSegments(), - IsClosed = false - } - } + Figures = new PathFigures(){figure} }; } diff --git a/Modules/AvaloniaGraph/Controls/ConnectionItem.cs b/Modules/AvaloniaGraph/Controls/ConnectionItem.cs index a1d083528..49d9af9e8 100644 --- a/Modules/AvaloniaGraph/Controls/ConnectionItem.cs +++ b/Modules/AvaloniaGraph/Controls/ConnectionItem.cs @@ -9,7 +9,7 @@ namespace AvaloniaGraph.Controls; public class ConnectionItem : ListBoxItem { public static readonly StyledProperty TopLeftPositionProperty = - AvaloniaProperty.Register(nameof(TopLeftPosition), + AvaloniaProperty.Register(nameof(TopLeftPosition), defaultBindingMode: BindingMode.TwoWay); private GraphControl? ParentGraphControl => this.FindAncestorOfType(); diff --git a/Modules/AvaloniaGraph/Controls/ConnectionsContainer.cs b/Modules/AvaloniaGraph/Controls/ConnectionsContainer.cs index 9c01643a3..dbe64bc48 100644 --- a/Modules/AvaloniaGraph/Controls/ConnectionsContainer.cs +++ b/Modules/AvaloniaGraph/Controls/ConnectionsContainer.cs @@ -5,15 +5,17 @@ namespace AvaloniaGraph.Controls; -public class ConnectionsContainer : ListBox, IStyleable +public class ConnectionsContainer : ListBox { - Type IStyleable.StyleKey => typeof(ListBox); + protected override Type StyleKeyOverride => typeof(ListBox); - protected override IItemContainerGenerator CreateItemContainerGenerator() + protected override Control CreateContainerForItemOverride(object? item, int index, object? recycleKey) { - return new ItemContainerGenerator( - this, - ContentControl.ContentProperty, - ContentControl.ContentTemplateProperty); + return new ConnectionItem(); + } + + protected override bool NeedsContainerOverride(object? item, int index, out object? recycleKey) + { + return this.NeedsContainer(item, out recycleKey); } } \ No newline at end of file diff --git a/Modules/AvaloniaGraph/Controls/ConnectorItem.cs b/Modules/AvaloniaGraph/Controls/ConnectorItem.cs index aeb564daf..8a2b920cb 100644 --- a/Modules/AvaloniaGraph/Controls/ConnectorItem.cs +++ b/Modules/AvaloniaGraph/Controls/ConnectorItem.cs @@ -79,7 +79,10 @@ private void UpdatePosition() centerPoint = new Point(0, Bounds.Height / 2); else if (attachMode == ConnectorAttachMode.Right) centerPoint = new Point(Bounds.Width, Bounds.Height / 2); - Position = this.TranslatePoint(centerPoint, ParentCanvas) ?? centerPoint; + if (ParentCanvas == null) + SetCurrentValue(PositionProperty, centerPoint); + else + SetCurrentValue(PositionProperty, this.TranslatePoint(centerPoint, ParentCanvas) ?? centerPoint); } #region Dependency properties diff --git a/Modules/AvaloniaGraph/Controls/Events/ConnectionDragCompletedEventArgs.cs b/Modules/AvaloniaGraph/Controls/Events/ConnectionDragCompletedEventArgs.cs index 0eabcfa15..5df330aef 100644 --- a/Modules/AvaloniaGraph/Controls/Events/ConnectionDragCompletedEventArgs.cs +++ b/Modules/AvaloniaGraph/Controls/Events/ConnectionDragCompletedEventArgs.cs @@ -6,7 +6,7 @@ namespace AvaloniaGraph.Controls; public class ConnectionDragCompletedEventArgs : ConnectionDragEventArgs { public ConnectionDragCompletedEventArgs(RoutedEvent routedEvent, - IInteractive? source, + Interactive? source, GraphNodeItemView? elementItem, object connection, ConnectorItem sourceConnectorItem, diff --git a/Modules/AvaloniaGraph/Controls/Events/ConnectionDragEventArgs.cs b/Modules/AvaloniaGraph/Controls/Events/ConnectionDragEventArgs.cs index 647c7393d..6baab7689 100644 --- a/Modules/AvaloniaGraph/Controls/Events/ConnectionDragEventArgs.cs +++ b/Modules/AvaloniaGraph/Controls/Events/ConnectionDragEventArgs.cs @@ -4,24 +4,25 @@ namespace AvaloniaGraph.Controls; -public abstract class ConnectionDragEventArgs : PointerEventArgs +public abstract class ConnectionDragEventArgs : RoutedEventArgs { protected ConnectionDragEventArgs(RoutedEvent routedEvent, - IInteractive? source, + Interactive? source, GraphNodeItemView? elementItem, ConnectorItem sourceConnectorItem, - PointerEventArgs baseArgs) : base(routedEvent, source, - baseArgs.Pointer, - (baseArgs.Source as IVisual)?.VisualRoot, - baseArgs.GetPosition(null), - baseArgs.Timestamp, - baseArgs.GetCurrentPoint(null).Properties, - baseArgs.KeyModifiers) + PointerEventArgs baseArgs) : base(routedEvent, source) { ElementItem = elementItem; SourceConnector = sourceConnectorItem; + BaseArgs = baseArgs; } + public PointerEventArgs BaseArgs { get; } public GraphNodeItemView? ElementItem { get; } public ConnectorItem SourceConnector { get; } + + + public IPointer Pointer => BaseArgs.Pointer; + public ulong Timestamp => BaseArgs.Timestamp; + public KeyModifiers KeyModifiers => BaseArgs.KeyModifiers; } \ No newline at end of file diff --git a/Modules/AvaloniaGraph/Controls/Events/ConnectionDraggingEventArgs.cs b/Modules/AvaloniaGraph/Controls/Events/ConnectionDraggingEventArgs.cs index c2d944bf7..70d304dcc 100644 --- a/Modules/AvaloniaGraph/Controls/Events/ConnectionDraggingEventArgs.cs +++ b/Modules/AvaloniaGraph/Controls/Events/ConnectionDraggingEventArgs.cs @@ -6,7 +6,7 @@ namespace AvaloniaGraph.Controls; public class ConnectionDraggingEventArgs : ConnectionDragEventArgs { public ConnectionDraggingEventArgs(RoutedEvent routedEvent, - IInteractive? source, + Interactive? source, GraphNodeItemView? elementItem, object connection, ConnectorItem connectorItem, diff --git a/Modules/AvaloniaGraph/Controls/Events/ConnectorItemBaseEventArgs.cs b/Modules/AvaloniaGraph/Controls/Events/ConnectorItemBaseEventArgs.cs index 82e97f90d..967c41efa 100644 --- a/Modules/AvaloniaGraph/Controls/Events/ConnectorItemBaseEventArgs.cs +++ b/Modules/AvaloniaGraph/Controls/Events/ConnectorItemBaseEventArgs.cs @@ -4,21 +4,19 @@ namespace AvaloniaGraph.Controls; -public abstract class ConnectorItemBaseEventArgs : PointerEventArgs +public abstract class ConnectorItemBaseEventArgs : RoutedEventArgs { public ConnectorItemBaseEventArgs(RoutedEvent routedEvent, - IInteractive? source, + Interactive? source, PointerEventArgs baseArgs) : - base(routedEvent, source, - baseArgs.Pointer, - (baseArgs.Source as IVisual)?.VisualRoot, - baseArgs.GetPosition(null), - baseArgs.Timestamp, - baseArgs.GetCurrentPoint(null).Properties, - baseArgs.KeyModifiers) + base(routedEvent, source) { BaseArgs = baseArgs; } public PointerEventArgs BaseArgs { get; } + + public IPointer Pointer => BaseArgs.Pointer; + public ulong Timestamp => BaseArgs.Timestamp; + public KeyModifiers KeyModifiers => BaseArgs.KeyModifiers; } \ No newline at end of file diff --git a/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDragCompletedEventArgs.cs b/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDragCompletedEventArgs.cs index f3dda32d4..45d185473 100644 --- a/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDragCompletedEventArgs.cs +++ b/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDragCompletedEventArgs.cs @@ -6,7 +6,7 @@ namespace AvaloniaGraph.Controls; public class ConnectorItemDragCompletedEventArgs : ConnectorItemBaseEventArgs { public ConnectorItemDragCompletedEventArgs(RoutedEvent routedEvent, - IInteractive? source, PointerEventArgs baseArgs) : base(routedEvent, source, baseArgs) + Interactive? source, PointerEventArgs baseArgs) : base(routedEvent, source, baseArgs) { } } diff --git a/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDraggingEventArgs.cs b/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDraggingEventArgs.cs index 6add1d4da..41b66aff6 100644 --- a/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDraggingEventArgs.cs +++ b/Modules/AvaloniaGraph/Controls/Events/ConnectorItemDraggingEventArgs.cs @@ -6,7 +6,7 @@ namespace AvaloniaGraph.Controls; public class ConnectorItemDraggingEventArgs : ConnectorItemBaseEventArgs { public ConnectorItemDraggingEventArgs(RoutedEvent routedEvent, - IInteractive? source, + Interactive? source, double horizontalChange, double verticalChange, PointerEventArgs baseArgs) : diff --git a/Modules/AvaloniaGraph/Controls/GraphControl.cs b/Modules/AvaloniaGraph/Controls/GraphControl.cs index 80634417f..25a0f7239 100644 --- a/Modules/AvaloniaGraph/Controls/GraphControl.cs +++ b/Modules/AvaloniaGraph/Controls/GraphControl.cs @@ -29,16 +29,16 @@ public GraphControl() public IList SelectedElements => nodesContainer?.SelectedItems ?? new List(); - public IEnumerable ElementsView => nodesContainer?.Items ?? Enumerable.Empty(); + public IEnumerable ElementsView => nodesContainer?.ItemsSource ?? Array.Empty(); public event EventHandler? SelectionChanged; protected override void OnApplyTemplate(TemplateAppliedEventArgs e) { - nodesContainer = (NodesContainer)e.NameScope.Find("PART_ElementItemsControl"); + nodesContainer = e.NameScope.Get("PART_ElementItemsControl"); nodesContainer.SelectionChanged += OnNodesContainerSelectChanged; - connectionsContainer = (ConnectionsContainer)e.NameScope.Find("PART_ConnectionItemsControl"); + connectionsContainer = e.NameScope.Get("PART_ConnectionItemsControl"); connectionsContainer.SelectionChanged += OnConnectionsContainerSelectChanged; base.OnApplyTemplate(e); @@ -55,20 +55,20 @@ private void OnNodesContainerSelectChanged(object? sender, SelectionChangedEvent //if (e.AddedItems.Count > 0) // connectionItemsControl.SelectedItem = null; if (SelectionChanged != null) - SelectionChanged(this, new SelectionChangedEventArgs(e.RoutedEvent, e.RemovedItems, e.AddedItems)); + SelectionChanged(this, new SelectionChangedEventArgs(e.RoutedEvent!, e.RemovedItems, e.AddedItems)); } protected override void OnPointerPressed(PointerPressedEventArgs e) { - nodesContainer!.SelectedItems.Clear(); - connectionsContainer!.SelectedItems.Clear(); + nodesContainer!.SelectedItems!.Clear(); + connectionsContainer!.SelectedItems!.Clear(); base.OnPointerPressed(e); } public int GetMaxZIndex() { - return nodesContainer!.ItemContainerGenerator.Containers - .Select(elementItem => ((GraphNodeItemView)elementItem.ContainerControl).ZIndex) + return nodesContainer!.ItemsView.Select(x => nodesContainer.ContainerFromItem(x!)) + .Select(elementItem => ((GraphNodeItemView)elementItem!.Parent!).ZIndex) .Concat(new[] { 0 }) .Max(); } @@ -115,9 +115,9 @@ public IList SelectedConnections AvaloniaProperty.Register( nameof(ConnectionItemTemplate)); - public DataTemplate ConnectionItemTemplate + public IDataTemplate ConnectionItemTemplate { - get => (DataTemplate)GetValue(ConnectionItemTemplateProperty); + get => (IDataTemplate)GetValue(ConnectionItemTemplateProperty); set => SetValue(ConnectionItemTemplateProperty, value); } diff --git a/Modules/AvaloniaGraph/Controls/GraphNodeItemView.cs b/Modules/AvaloniaGraph/Controls/GraphNodeItemView.cs index 50e7f8fc6..aff23ebf6 100644 --- a/Modules/AvaloniaGraph/Controls/GraphNodeItemView.cs +++ b/Modules/AvaloniaGraph/Controls/GraphNodeItemView.cs @@ -20,7 +20,7 @@ public void BringToFront() return; var maxZ = ParentGraphControl.GetMaxZIndex(); - ZIndex = maxZ + 1; + SetCurrentValue(ZIndexProperty, maxZ + 1); } #region Dependency properties @@ -52,16 +52,16 @@ public bool IsDragging get => GetValue(IsDraggingProperty); set => SetValue(IsDraggingProperty, value); } - - public static readonly StyledProperty ZIndexProperty = AvaloniaProperty.Register( - nameof(ZIndex), - defaultBindingMode: BindingMode.TwoWay); - - public int ZIndex - { - get => GetValue(ZIndexProperty); - set => SetValue(ZIndexProperty, value); - } + // + // public static readonly StyledProperty ZIndexProperty = AvaloniaProperty.Register( + // nameof(ZIndex), + // defaultBindingMode: BindingMode.TwoWay); + // + // public int ZIndex + // { + // get => GetValue(ZIndexProperty); + // set => SetValue(ZIndexProperty, value); + // } #endregion @@ -76,7 +76,7 @@ private void DoSelection() return; ParentGraphControl.ClearSelection(); - IsSelected = true; + SetCurrentValue(IsSelectedProperty, true); } protected override void OnPointerPressed(PointerPressedEventArgs e) @@ -124,7 +124,7 @@ protected override void OnPointerMoved(PointerEventArgs e) { startX = X; startY = Y; - IsDragging = true; + SetCurrentValue(IsDraggingProperty, true); //CaptureMouse(); } @@ -140,7 +140,7 @@ protected override void OnPointerReleased(PointerReleasedEventArgs e) if (IsDragging) //ReleaseMouseCapture(); - IsDragging = false; + SetCurrentValue(IsDraggingProperty, false); } e.Handled = true; diff --git a/Modules/AvaloniaGraph/Controls/NodesContainer.cs b/Modules/AvaloniaGraph/Controls/NodesContainer.cs index 38b0dd060..adeb57c2c 100644 --- a/Modules/AvaloniaGraph/Controls/NodesContainer.cs +++ b/Modules/AvaloniaGraph/Controls/NodesContainer.cs @@ -5,15 +5,17 @@ namespace AvaloniaGraph.Controls; -public class NodesContainer : ListBox, IStyleable +public class NodesContainer : ListBox { - Type IStyleable.StyleKey => typeof(ListBox); + protected override Type StyleKeyOverride => typeof(ListBox); + + protected override Control CreateContainerForItemOverride(object? item, int index, object? recycleKey) + { + return new GraphNodeItemView(); + } - protected override IItemContainerGenerator CreateItemContainerGenerator() + protected override bool NeedsContainerOverride(object? item, int index, out object? recycleKey) { - return new ItemContainerGenerator( - this, - ContentControl.ContentProperty, - ContentControl.ContentTemplateProperty); + return this.NeedsContainer(item, out recycleKey); } } \ No newline at end of file diff --git a/Modules/AvaloniaGraph/Style.axaml b/Modules/AvaloniaGraph/Style.axaml index b33eb66c2..a575f3e4c 100644 --- a/Modules/AvaloniaGraph/Style.axaml +++ b/Modules/AvaloniaGraph/Style.axaml @@ -94,8 +94,8 @@ @@ -106,12 +106,11 @@ @@ -119,9 +118,9 @@ @@ -132,12 +131,11 @@ diff --git a/Modules/CrashReport/App.axaml b/Modules/CrashReport/App.axaml index a93aefb40..277fcb43a 100644 --- a/Modules/CrashReport/App.axaml +++ b/Modules/CrashReport/App.axaml @@ -2,15 +2,14 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:crashReport="clr-namespace:CrashReport" x:Class="CrashReport.App" + RequestedThemeVariant="Default" > - - - + \ No newline at end of file diff --git a/Modules/CrashReport/AvaloniaViewLocator.cs b/Modules/CrashReport/AvaloniaViewLocator.cs index 500d736d0..5133d6e03 100644 --- a/Modules/CrashReport/AvaloniaViewLocator.cs +++ b/Modules/CrashReport/AvaloniaViewLocator.cs @@ -1,12 +1,13 @@ using System; using Avalonia.Controls; using Avalonia.Controls.Templates; +using Avalonia.Markup.Xaml.Templates; namespace CrashReport; public class AvaloniaViewLocator : IDataTemplate { - public IControl Build(object? param) + public Control? Build(object? param) { if (param == null) return new TextBlock(){Text = "Null model"}; @@ -25,6 +26,6 @@ public IControl Build(object? param) public bool Match(object? data) { - return data is not IControl && data != null && data is not string; + return data is not Control && data != null && data is not string; } } \ No newline at end of file diff --git a/Modules/CrashReport/CrashReport.csproj b/Modules/CrashReport/CrashReport.csproj index b44d42286..76419674b 100644 --- a/Modules/CrashReport/CrashReport.csproj +++ b/Modules/CrashReport/CrashReport.csproj @@ -1,9 +1,9 @@  WinExe - net7.0 + net8.0 enable - nullable + $(WarningsAsErrors),nullable Icon.ico true app.manifest @@ -17,6 +17,7 @@ + diff --git a/Modules/CrashReport/CrashReportView.axaml b/Modules/CrashReport/CrashReportView.axaml index 1c2bfd8e5..9b6f13954 100644 --- a/Modules/CrashReport/CrashReportView.axaml +++ b/Modules/CrashReport/CrashReportView.axaml @@ -15,15 +15,16 @@ - The editor has crashed 😭 + The editor has crashed 😭 - I am very sorry, but due to some internal problem, the editor has crashed. This report will help me prevent it in the future, so if it happens again, please report the error on github. + I am very sorry, but due to some internal problem, the editor has crashed. This report will help me prevent it in the future, so consider sending the report to the authors. Crash report: + - + diff --git a/Modules/CrashReport/CrashReportViewModel.cs b/Modules/CrashReport/CrashReportViewModel.cs index e96683d1b..9da5b477e 100644 --- a/Modules/CrashReport/CrashReportViewModel.cs +++ b/Modules/CrashReport/CrashReportViewModel.cs @@ -1,10 +1,24 @@ using System; +using System.Collections.Generic; +using System.ComponentModel; using System.IO; +using System.Net.Http; +using System.Net.Http.Headers; +using System.Net.Http.Json; +using System.Runtime.CompilerServices; +using System.Windows.Input; +using AsyncAwaitBestPractices.MVVM; +using Newtonsoft.Json; +using WDE.Common.Factories; +using WDE.Common.Services; +using WDE.Common.Utils; namespace CrashReport; -public class CrashReportViewModel +public class CrashReportViewModel : INotifyPropertyChanged { + private readonly IApplicationReleaseConfiguration configuration; + private readonly IHttpClientFactory clientFactory; public string CrashReport { get; } private static string APPLICATION_FOLDER = "WoWDatabaseEditor"; @@ -14,12 +28,63 @@ public class CrashReportViewModel private static string WDE_DATA_FOLDER => Path.Join(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), APPLICATION_FOLDER); private static string FATAL_LOG_FILE => Path.Join(WDE_DATA_FOLDER, LOG_FILE); private static string FATAL_LOG_FILE_OLD => Path.Join(WDE_DATA_FOLDER, LOG_FILE_OLD); - - public CrashReportViewModel() + + private bool reportSent; + public AsyncCommand UploadReportCommand { get; } + + public string SendCrashLogButtonText { get; private set; } = "Send the crash log"; + + public CrashReportViewModel(IApplicationReleaseConfiguration configuration, IHttpClientFactory clientFactory) { + this.configuration = configuration; + this.clientFactory = clientFactory; if (File.Exists(FATAL_LOG_FILE)) CrashReport = File.ReadAllText(FATAL_LOG_FILE); else + { CrashReport = "(crash log not found :/)"; + reportSent = true; + } + + var server = configuration.GetString("UPDATE_SERVER"); + if (string.IsNullOrEmpty(server)) + reportSent = true; + + UploadReportCommand = new AsyncCommand(async () => + { + reportSent = true; + UploadReportCommand?.RaiseCanExecuteChanged(); + + var httpClient = clientFactory.Factory(); + try + { + await httpClient.PostAsync(Path.Join(server, "Log", "Send"), new StringContent(JsonConvert.SerializeObject(new + { + log = CrashReport + }), new MediaTypeHeaderValue("application/json"))); + SendCrashLogButtonText = "Crash log sent!"; + OnPropertyChanged(nameof(SendCrashLogButtonText)); + } + catch (Exception e) + { + SendCrashLogButtonText = "Couldn't sent the crashlog: " + e.Message; + OnPropertyChanged(nameof(SendCrashLogButtonText)); + } + }, _ => !reportSent); + } + + public event PropertyChangedEventHandler? PropertyChanged; + + protected virtual void OnPropertyChanged([CallerMemberName] string? propertyName = null) + { + PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName)); + } + + protected bool SetField(ref T field, T value, [CallerMemberName] string? propertyName = null) + { + if (EqualityComparer.Default.Equals(field, value)) return false; + field = value; + OnPropertyChanged(propertyName); + return true; } } \ No newline at end of file diff --git a/Modules/CrashReport/MainWindowViewModel.cs b/Modules/CrashReport/MainWindowViewModel.cs index 9ab18e7f9..4418e1118 100644 --- a/Modules/CrashReport/MainWindowViewModel.cs +++ b/Modules/CrashReport/MainWindowViewModel.cs @@ -30,15 +30,6 @@ public class MainWindowViewModel : ITaskProgress, INotifyPropertyChanged public MainWindowViewModel() { - if (GlobalApplication.Arguments.IsArgumentSet("crashed")) - { - MainContent = new CrashReportViewModel(); - } - else - { - MainContent = new NoCrashViewModel(); - } - RestartEditorWithConsoleCommand = new DelegateCommand(() => { var locator = new ProgramFinder(); @@ -75,8 +66,9 @@ public MainWindowViewModel() IFileSystem fs = new FileSystem(); var releaseConfig = new ApplicationReleaseConfiguration(fs); var applicationVersion = new ApplicationVersion(releaseConfig); + var httpFactory = new HttpClientFactory(applicationVersion); var updateService = new UpdateService(new UpdateServerDataProvider(releaseConfig), - new UpdateClientFactory(new HttpClientFactory(applicationVersion)), + new UpdateClientFactory(httpFactory), new ApplicationVersion(releaseConfig), new ApplicationImpl(), fs, @@ -91,6 +83,16 @@ public MainWindowViewModel() await updateService.DownloadLatestVersion(this); await updateService.CloseForUpdate(); }, () => updateService.CanCheckForUpdates()); + + + if (GlobalApplication.Arguments.IsArgumentSet("crashed")) + { + MainContent = new CrashReportViewModel(releaseConfig, httpFactory); + } + else + { + MainContent = new NoCrashViewModel(); + } } public TaskState State { get; set; } diff --git a/Modules/CrashReport/NoCrashView.axaml b/Modules/CrashReport/NoCrashView.axaml index 25e516f26..55abc81f0 100644 --- a/Modules/CrashReport/NoCrashView.axaml +++ b/Modules/CrashReport/NoCrashView.axaml @@ -15,10 +15,10 @@ - The editor has not crashed 🙂 + The editor has not crashed 🙂 - It looks like the editor hasn't crashed, it is just you who manually started this CrashReport app. You can use it to redownload the latest version. + It looks like the editor hasn't crashed, you just manually started this CrashReport app. You can use it to redownload the latest version. diff --git a/Modules/CrashReport/Program.cs b/Modules/CrashReport/Program.cs index dc383dee0..2e4f499f5 100644 --- a/Modules/CrashReport/Program.cs +++ b/Modules/CrashReport/Program.cs @@ -22,7 +22,9 @@ public static void Main(string[] args) private static void FixCurrentDirectory() { + #pragma warning disable IL3000 var path = Assembly.GetExecutingAssembly().Location; + #pragma warning restore IL3000 if (string.IsNullOrEmpty(path)) path = System.AppContext.BaseDirectory; var exePath = new FileInfo(path); @@ -34,6 +36,6 @@ private static void FixCurrentDirectory() public static AppBuilder BuildAvaloniaApp() => AppBuilder.Configure() .UsePlatformDetect() - //.WithInterFont() // ava11 + .WithInterFont() .LogToTrace(); } \ No newline at end of file diff --git a/Modules/WDE.AnniversaryInfo/Services/CommentPublisherService.cs b/Modules/WDE.AnniversaryInfo/Services/CommentPublisherService.cs index bfcc1f791..6429da42d 100644 --- a/Modules/WDE.AnniversaryInfo/Services/CommentPublisherService.cs +++ b/Modules/WDE.AnniversaryInfo/Services/CommentPublisherService.cs @@ -5,6 +5,7 @@ using System.Threading.Tasks; using Avalonia.Input; using Newtonsoft.Json; +using WDE.Common; using WDE.Common.Factories; using WDE.Common.Services; using WDE.Module.Attributes; @@ -47,7 +48,7 @@ public async Task Publish(string username, string text) } catch (Exception e) { - Console.WriteLine(e); + LOG.LogError(e); return CommentResult.InternetProblem; } } diff --git a/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml b/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml index f1f408dbb..74209285e 100644 --- a/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml +++ b/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml @@ -42,7 +42,7 @@ - + @@ -55,7 +55,7 @@ - + diff --git a/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml.cs b/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml.cs index f891e532f..ff01f5994 100644 --- a/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml.cs +++ b/Modules/WDE.AnniversaryInfo/Views/TimelineView.axaml.cs @@ -4,7 +4,7 @@ namespace WDE.AnniversaryInfo.Views; -public class TimelineView : UserControl +public partial class TimelineView : UserControl { public TimelineView() { diff --git a/Modules/WDE.AnniversaryInfo/WDE.AnniversaryInfo.csproj b/Modules/WDE.AnniversaryInfo/WDE.AnniversaryInfo.csproj index 553453bb0..131a34dc2 100644 --- a/Modules/WDE.AnniversaryInfo/WDE.AnniversaryInfo.csproj +++ b/Modules/WDE.AnniversaryInfo/WDE.AnniversaryInfo.csproj @@ -1,12 +1,12 @@ - net7.0 + net8.0 0.0.0.0 0.0.0.0 Debug;Release AnyCPU enable - nullable + $(WarningsAsErrors),nullable diff --git a/Modules/WDE.DatabaseDefinitionEditor/DatabaseDefinitionEditorModule.cs b/Modules/WDE.DatabaseDefinitionEditor/DatabaseDefinitionEditorModule.cs index b08b56a9d..9a1379bc0 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/DatabaseDefinitionEditorModule.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/DatabaseDefinitionEditorModule.cs @@ -1,4 +1,7 @@ -using WDE.Common.Windows; +using System; +using Avalonia; +using Avalonia.Markup.Xaml.Styling; +using WDE.Common.Windows; using WDE.DatabaseDefinitionEditor.ViewModels; using WDE.DatabaseDefinitionEditor.Views; using WDE.Module; @@ -14,5 +17,7 @@ public override void RegisterViews(IViewLocator viewLocator) viewLocator.Bind(); // table editor //viewLocator.Bind(); + + Application.Current!.Styles.Add(new StyleInclude(new Uri("resm:Styles?assembly=WDE.DatabaseDefinitionEditor")){Source = new Uri("avares://WDE.DatabaseDefinitionEditor/Themes/Generic.axaml")}); } } \ No newline at end of file diff --git a/Modules/WDE.DatabaseDefinitionEditor/ViewModels/DefinitionEditorViewModel.cs b/Modules/WDE.DatabaseDefinitionEditor/ViewModels/DefinitionEditorViewModel.cs index c71ec51c6..51ccfeaad 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/ViewModels/DefinitionEditorViewModel.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/ViewModels/DefinitionEditorViewModel.cs @@ -11,6 +11,7 @@ using Avalonia.Input; using Avalonia.Threading; using DynamicData.Binding; +using Microsoft.Extensions.Logging; using Newtonsoft.Json; using PropertyChanged.SourceGenerator; using WDE.Common.CoreVersion; @@ -420,7 +421,9 @@ private async Task RefreshPreview(DatabaseTableDefinitionJson? json, Defin var tf = tg.Fields[j]; var of = og.Fields[j]; if (tf != of) - {Console.WriteLine("At index " + i + "/" + j + " there is a diff");} + { + LOG.LogWarning("At index " + i + "/" + j + " there is a diff"); + } } } } diff --git a/Modules/WDE.DatabaseDefinitionEditor/ViewModels/SourceTreeTableDefinitionEditorProvider.cs b/Modules/WDE.DatabaseDefinitionEditor/ViewModels/SourceTreeTableDefinitionEditorProvider.cs index 47a2dcb7f..61ba5014e 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/ViewModels/SourceTreeTableDefinitionEditorProvider.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/ViewModels/SourceTreeTableDefinitionEditorProvider.cs @@ -3,6 +3,7 @@ using System.IO; using System.Linq; using Newtonsoft.Json; +using WDE.Common; using WDE.Common.CoreVersion; using WDE.DatabaseEditors.Data.Structs; using WDE.Module.Attributes; @@ -55,7 +56,7 @@ private bool TryAdd(DirectoryInfo? root) } catch (Exception e) { - Console.WriteLine(e); + LOG.LogError(e); } } diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnView.axaml index 824c1b9b7..32b4f2ea1 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnView.axaml @@ -30,7 +30,7 @@ - an actual database column - a button that opens the conditions editor - a special button which meaning can be customized (meta column)">Column type: - + @@ -93,7 +93,7 @@ This will generate a comment, it will substitute `thisString.COLUMN` with a stri Column type: + ItemsSource="{CompiledBinding Parent.Parent.Parent.MetaColumnFactory.Types}"> diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnsView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnsView.axaml index 9920e5d65..f36937004 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnsView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/ColumnsView.axaml @@ -23,7 +23,7 @@ Foreign tables: - + @@ -51,7 +51,7 @@ Background="{DynamicResource ThemeBackgroundBrush}" BorderThickness="{DynamicResource ThemeBorderThickness}" BorderBrush="{DynamicResource ThemeBorderMidBrush}" - Items="{CompiledBinding Groups}" + ItemsSource="{CompiledBinding Groups}" SelectedItem="{CompiledBinding SelectedColumnOrGroup}"> diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/CommandView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/CommandView.axaml index 320e1c4ec..53bdda6ac 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/CommandView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/CommandView.axaml @@ -33,7 +33,7 @@ - + diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/CommandsView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/CommandsView.axaml index 7101883d6..48d11d64a 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/CommandsView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/CommandsView.axaml @@ -14,7 +14,7 @@ - + diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/ConditionsView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/ConditionsView.axaml index fdb33f348..7eeba290a 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/ConditionsView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/ConditionsView.axaml @@ -53,7 +53,7 @@ - + diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseColumnCompletionBox.cs b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseColumnCompletionBox.cs index 72afbc434..ab25c7b77 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseColumnCompletionBox.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseColumnCompletionBox.cs @@ -16,12 +16,10 @@ namespace WDE.DatabaseDefinitionEditor.Views.Controls; -public class DatabaseColumnCompletionBox : CompletionComboBox, IStyleable +public class DatabaseColumnCompletionBox : CompletionComboBox { - Type IStyleable.StyleKey => typeof(CompletionComboBox); - public static readonly StyledProperty TableNameProperty = AvaloniaProperty.Register(nameof(TableName)); - //protected override Type StyleKeyOverride => typeof(CompletionComboBox); // avalonia 11 + protected override Type StyleKeyOverride => typeof(CompletionComboBox); public string? TableName { diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCommandCompletionBox.cs b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCommandCompletionBox.cs index 4c21edc32..39c020b73 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCommandCompletionBox.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCommandCompletionBox.cs @@ -13,9 +13,9 @@ namespace WDE.DatabaseDefinitionEditor.Views.Controls; -public class DatabaseTableCommandCompletionBox : CompletionComboBox, IStyleable +public class DatabaseTableCommandCompletionBox : CompletionComboBox { - Type IStyleable.StyleKey => typeof(CompletionComboBox); + protected override Type StyleKeyOverride => typeof(CompletionComboBox); public string CommandId { diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCompletionBox.cs b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCompletionBox.cs index 9cb98dc66..cb5ee8ff1 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCompletionBox.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/DatabaseTableCompletionBox.cs @@ -15,9 +15,9 @@ namespace WDE.DatabaseDefinitionEditor.Views.Controls; -public class DatabaseTableCompletionBox : CompletionComboBox, IStyleable +public class DatabaseTableCompletionBox : CompletionComboBox { - Type IStyleable.StyleKey => typeof(CompletionComboBox); + protected override Type StyleKeyOverride => typeof(CompletionComboBox); public static readonly DirectProperty TableNameProperty = AvaloniaProperty.RegisterDirect(nameof(TableName), o => o.TableName, (o, v) => o.TableName = v, defaultBindingMode: BindingMode.TwoWay); @@ -74,7 +74,7 @@ static DatabaseTableCompletionBox() private CancellationTokenSource? cts; public static readonly StyledProperty CanSelectEmptyProperty = - AvaloniaProperty.Register(nameof(CanSelectEmpty)); + AvaloniaProperty.Register(nameof(CanSelectEmpty)); public static readonly StyledProperty DatabaseProperty = AvaloniaProperty.Register("Database"); diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/KeyBindingBox.cs b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/KeyBindingBox.cs index f4f3c3370..ff31b59ac 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/KeyBindingBox.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/KeyBindingBox.cs @@ -10,9 +10,9 @@ namespace WDE.DatabaseDefinitionEditor.Views.Controls; -public class TextBoxWithNoInput : TextBox, IStyleable +public class TextBoxWithNoInput : TextBox { - Type IStyleable.StyleKey => typeof(TextBox); + protected override Type StyleKeyOverride => typeof(TextBox); protected override void OnTextInput(TextInputEventArgs e) { @@ -85,7 +85,7 @@ private void Handler(object? sender, KeyEventArgs e) else if (e.Key == Key.LeftAlt || e.Key == Key.RightAlt) modifier &= ~KeyModifiers.Alt; - KeyGesture = new KeyGesture(e.Key, modifier); + SetCurrentValue(KeyGestureProperty, new KeyGesture(e.Key, modifier)); e.Handled = true; } } \ No newline at end of file diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/NullableDatabaseColumnCompletionBox.cs b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/NullableDatabaseColumnCompletionBox.cs index c7c1d59f6..b67ae61fa 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/NullableDatabaseColumnCompletionBox.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/NullableDatabaseColumnCompletionBox.cs @@ -69,7 +69,7 @@ private void UpdateIsNull() return; inEvent = true; - IsNotNull = !string.IsNullOrWhiteSpace(columnName); + SetCurrentValue(IsNotNullProperty, !string.IsNullOrWhiteSpace(columnName)); inEvent = false; } diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/OnReleaseListBoxItem.cs b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/OnReleaseListBoxItem.cs index 2cdeaabfc..4d768efa9 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/OnReleaseListBoxItem.cs +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/Controls/OnReleaseListBoxItem.cs @@ -2,16 +2,21 @@ using Avalonia; using Avalonia.Controls; using Avalonia.Input; +using Avalonia.Interactivity; using Avalonia.Styling; using Avalonia.VisualTree; namespace WDE.DatabaseDefinitionEditor.Views.Controls; -// note: in avalonia 11 this is not required. -// this changes the listbox to update the selection on pointer release rather than pointer press -public class OnReleaseListBox : ListBox, IStyleable +// this changes the listbox to update the selection on pointer release rather than pointer press (originally handled in ListBoxItem) +public class OnReleaseListBox : ListBox { - Type IStyleable.StyleKey => typeof(ListBox); + protected override Type StyleKeyOverride => typeof(ListBox); + + static OnReleaseListBox() + { + PointerPressedEvent.AddClassHandler((x, e) => x.OnPointerPressed(e), RoutingStrategies.Tunnel); + } protected override void OnPointerPressed(PointerPressedEventArgs e) { @@ -21,7 +26,7 @@ protected override void OnPointerPressed(PointerPressedEventArgs e) protected override void OnPointerReleased(Avalonia.Input.PointerReleasedEventArgs e) { base.OnPointerReleased(e); - if (e.Source is IVisual source) + if (e.Source is Visual source) { var point = e.GetCurrentPoint(source); @@ -30,8 +35,8 @@ protected override void OnPointerReleased(Avalonia.Input.PointerReleasedEventArg e.Handled = UpdateSelectionFromEventSource( e.Source, true, - e.KeyModifiers.HasAllFlags(KeyModifiers.Shift), - e.KeyModifiers.HasAllFlags(KeyModifiers.Control), + e.KeyModifiers.HasFlagFast(KeyModifiers.Shift), + e.KeyModifiers.HasFlagFast(KeyModifiers.Control), point.Properties.IsRightButtonPressed); } } diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionEditorView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionEditorView.axaml index c6f23a7ab..64880c3cc 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionEditorView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionEditorView.axaml @@ -17,7 +17,7 @@ Definitions: - @@ -34,7 +34,7 @@ Pick a table to import: - + @@ -45,15 +45,9 @@ - - - - - - - diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionToolView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionToolView.axaml index fe31848e1..5d3dc1204 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionToolView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionToolView.axaml @@ -20,7 +20,7 @@ Connected database tables: - @@ -43,7 +43,7 @@ All provided table definitions: - @@ -70,9 +70,9 @@ - + - + diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionView.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionView.axaml index 85c3a4349..d8bba96cd 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionView.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/DefinitionView.axaml @@ -13,7 +13,7 @@ x:DataType="definitionEditor:DefinitionViewModel"> Database: - + Table name: @@ -40,7 +40,7 @@ Table record mode: - + Friendly name: @@ -66,7 +66,7 @@ MultiRecord and SingleRow can be used for tables with multi (or single) column p IsVisible="{CompiledBinding IsSingleRow, Converter={x:Static BoolConverters.Not}}" /> Is only conditions table: - + Flags: @@ -89,7 +89,7 @@ MultiRecord and SingleRow can be used for tables with multi (or single) column p - + @@ -142,7 +142,7 @@ I.e. user looks for sound = 1423 and it finds row in the creature_text table wit - + @@ -169,7 +169,7 @@ I.e. user looks for sound = 1423 and it finds row in the creature_text table wit Auto key value: - diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/ForeignTables.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/ForeignTables.axaml index 9f9b368e3..fa027d852 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/ForeignTables.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/ForeignTables.axaml @@ -16,7 +16,7 @@ - + @@ -51,7 +51,7 @@ - + diff --git a/Modules/WDE.DatabaseDefinitionEditor/Views/TemplateModePreview.axaml b/Modules/WDE.DatabaseDefinitionEditor/Views/TemplateModePreview.axaml index 700911838..1af2574b9 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/Views/TemplateModePreview.axaml +++ b/Modules/WDE.DatabaseDefinitionEditor/Views/TemplateModePreview.axaml @@ -8,7 +8,7 @@ mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" x:DataType="definitionEditor:DefinitionViewModel" x:Class="WDE.DatabaseDefinitionEditor.Views.TemplateModePreview"> - + @@ -27,7 +27,7 @@ VerticalAlignment="Center"/> - + diff --git a/Modules/WDE.DatabaseDefinitionEditor/WDE.DatabaseDefinitionEditor.csproj b/Modules/WDE.DatabaseDefinitionEditor/WDE.DatabaseDefinitionEditor.csproj index 3e4ad8f4f..0c216f9ed 100644 --- a/Modules/WDE.DatabaseDefinitionEditor/WDE.DatabaseDefinitionEditor.csproj +++ b/Modules/WDE.DatabaseDefinitionEditor/WDE.DatabaseDefinitionEditor.csproj @@ -1,12 +1,12 @@ - net7.0 + net8.0 Library false Debug;Release AnyCPU enable - nullable + $(WarningsAsErrors),nullable ..\bin\$(Configuration)\ diff --git a/Modules/WDE.DatabaseEditors.Test/Services/QueryParserServiceTests.cs b/Modules/WDE.DatabaseEditors.Test/Services/QueryParserServiceTests.cs index 5696cfda7..4b40a3028 100644 --- a/Modules/WDE.DatabaseEditors.Test/Services/QueryParserServiceTests.cs +++ b/Modules/WDE.DatabaseEditors.Test/Services/QueryParserServiceTests.cs @@ -19,7 +19,7 @@ namespace WDE.DatabaseEditors.Test.Services; public class QueryParserServiceTests { - private IQueryParserService parserService; + private IQueryParserService parserService = null!; [SetUp] public void Setup() @@ -67,7 +67,7 @@ public void Setup() dataLoader.Load(default!, default, default, default, default) .ReturnsForAnyArgs( - Task.FromResult(new DatabaseTableData(definition, new List() + Task.FromResult(new DatabaseTableData(definition, new List() { new DatabaseEntity(true, new DatabaseKey(0, 18675), new Dictionary(), null) }))); diff --git a/Modules/WDE.DatabaseEditors.Test/WDE.DatabaseEditors.Test.csproj b/Modules/WDE.DatabaseEditors.Test/WDE.DatabaseEditors.Test.csproj index dfc218580..d9e484b9b 100644 --- a/Modules/WDE.DatabaseEditors.Test/WDE.DatabaseEditors.Test.csproj +++ b/Modules/WDE.DatabaseEditors.Test/WDE.DatabaseEditors.Test.csproj @@ -1,7 +1,7 @@ - net7.0 + net8.0 enable false @@ -10,9 +10,9 @@ - - - + + + diff --git a/Modules/WDE.Debugger.Test/Services/DebuggerServiceTests.cs b/Modules/WDE.Debugger.Test/Services/DebuggerServiceTests.cs index 3977b0db0..b4dc05105 100644 --- a/Modules/WDE.Debugger.Test/Services/DebuggerServiceTests.cs +++ b/Modules/WDE.Debugger.Test/Services/DebuggerServiceTests.cs @@ -647,11 +647,10 @@ public async Task EditorConnected_EventFired_ShouldSynchronizeAppropriateDebugPo }); bool exceptionSynchronizationAttempted = false; - source3.Synchronizer.Synchronize(exceptionDebugPointId).Returns(async _ => + source3.Synchronizer.Synchronize(exceptionDebugPointId).Returns>(async _ => { exceptionSynchronizationAttempted = true; throw new Exception("Synchronization failed."); - return SynchronizationResult.Ok; }); // Act diff --git a/Modules/WDE.Debugger.Test/WDE.Debugger.Test.csproj b/Modules/WDE.Debugger.Test/WDE.Debugger.Test.csproj index 839f84bd3..b60ef4de6 100644 --- a/Modules/WDE.Debugger.Test/WDE.Debugger.Test.csproj +++ b/Modules/WDE.Debugger.Test/WDE.Debugger.Test.csproj @@ -1,9 +1,9 @@ - net7.0 + net8.0 enable - nullable + $(WarningsAsErrors),nullable false @@ -11,9 +11,9 @@ - - - + + + diff --git a/Modules/WDE.Debugger/Services/DebuggerService.cs b/Modules/WDE.Debugger/Services/DebuggerService.cs index a69fa3baf..cde890eef 100644 --- a/Modules/WDE.Debugger/Services/DebuggerService.cs +++ b/Modules/WDE.Debugger/Services/DebuggerService.cs @@ -6,6 +6,7 @@ using Newtonsoft.Json; using Newtonsoft.Json.Linq; using Prism.Events; +using WDE.Common; using WDE.Common.Debugging; using WDE.Common.Services; using WDE.Common.Utils; @@ -158,7 +159,7 @@ void LoadSnapshot(DebugPointSnapshot snapshot) { if (!sourcesByKey.TryGetValue(snapshot.SourceName, out var source)) { - Console.WriteLine("Couldn't deserialize debug point with key: " + snapshot.SourceName); + LOG.LogWarning("Couldn't deserialize debug point with key: " + snapshot.SourceName); return; } @@ -193,13 +194,13 @@ void LoadSnapshot(DebugPointSnapshot snapshot) } catch (Exception e) { - Console.WriteLine("Couldn't load a snapshot: " + e); + LOG.LogError(e, message: "Couldn't load a snapshot"); } } } catch (Exception e) { - Console.WriteLine("Couldn't load snapshots: " + e); + LOG.LogError(e, message: "Couldn't load snapshots"); } } @@ -256,7 +257,7 @@ private async Task SynchronizePointAndUpdateState(DebugPointModel debug) else throw new ArgumentOutOfRangeException(nameof(result)); } - catch (OperationCanceledException e) + catch (OperationCanceledException) { debug.State = BreakpointState.Pending; } @@ -669,11 +670,11 @@ public async Task Synchronize(DebugPointId id) await debug.Source.Synchronizer.Delete(id); debug.State = BreakpointState.Synced; } - catch (OperationCanceledException e) + catch (OperationCanceledException) { debug.State = BreakpointState.Pending; } - catch (Exception e) + catch (Exception) { debug.State = BreakpointState.SynchronizationError; throw; @@ -692,12 +693,12 @@ public async Task Synchronize(DebugPointId id) await debug.Source.Synchronizer.Delete(id); RemoveDebugPointInternal(id); } - catch (DebuggingFeaturesDisabledException e) + catch (DebuggingFeaturesDisabledException) { RemoveDebugPointInternal(id); throw; } - catch (Exception e) + catch (Exception) { debug.State = BreakpointState.SynchronizationError; CallDebugPointChangedInternal(id); diff --git a/Modules/WDE.Debugger/Views/Inspector/DebugPointsInspectorToolBar.axaml b/Modules/WDE.Debugger/Views/Inspector/DebugPointsInspectorToolBar.axaml index fcc51e14b..78a94bad5 100644 --- a/Modules/WDE.Debugger/Views/Inspector/DebugPointsInspectorToolBar.axaml +++ b/Modules/WDE.Debugger/Views/Inspector/DebugPointsInspectorToolBar.axaml @@ -10,7 +10,7 @@ - + diff --git a/Modules/WDE.SqlWorkbench/Views/DumpView.axaml b/Modules/WDE.SqlWorkbench/Views/DumpView.axaml index c232e9a2b..4f830638c 100644 --- a/Modules/WDE.SqlWorkbench/Views/DumpView.axaml +++ b/Modules/WDE.SqlWorkbench/Views/DumpView.axaml @@ -36,8 +36,7 @@ - @@ -76,7 +75,7 @@ - + diff --git a/Modules/WDE.SqlWorkbench/Views/SelectSingleTableView.axaml.cs b/Modules/WDE.SqlWorkbench/Views/SelectSingleTableView.axaml.cs index 8117d5cd5..f421d47a2 100644 --- a/Modules/WDE.SqlWorkbench/Views/SelectSingleTableView.axaml.cs +++ b/Modules/WDE.SqlWorkbench/Views/SelectSingleTableView.axaml.cs @@ -1,6 +1,7 @@ using System; using Avalonia; using Avalonia.Controls; +using Avalonia.Input; using Avalonia.Interactivity; using Avalonia.Markup.Xaml; using Avalonia.Threading; @@ -87,7 +88,7 @@ private void VirtualizedVeryFastTableView_OnValueUpdateRequest(string newValue) } } - private void InputElement_OnDoubleTapped(object? sender, RoutedEventArgs e) + private void InputElement_OnDoubleTapped(object? sender, TappedEventArgs e) { if (DataContext is SelectSingleTableViewModel vm) { diff --git a/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchToolBar.axaml b/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchToolBar.axaml index 446de3b98..ba63720b8 100644 --- a/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchToolBar.axaml +++ b/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchToolBar.axaml @@ -40,7 +40,7 @@ - + diff --git a/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchView.axaml b/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchView.axaml index 294a8da74..df625a72e 100644 --- a/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchView.axaml +++ b/Modules/WDE.SqlWorkbench/Views/SqlWorkbenchView.axaml @@ -41,7 +41,7 @@ Change connection for this tab: - +